# The ins and outs of the PS/2 machines' video systems

# **PS/2 Video Programming**

### **Richard Wilton**

The IBM PS/2 series introduces two new video subsystems, the multicolor graphics array (MCGA) and video graphics array (VGA). This article is an overview of the MCGA and the VGA from a programmer's point of view. If you are already familiar with older video adapters, such as the CGA, EGA, or Hercules cards, this article will point out the similarities and differences between the PS/2 video subsystems and previous IBM video adapters. If you are new to video hardware programming, you can use the examples in this article as a focus for further exploration of the PS/2 hardware.

Unlike the IBM PC, XT, and AT, into which you must install a separate card that supports the necessary hardware to drive a video display, all the PS/2s are equipped with a built-in video subsystem on the motherboard. The Model 30 comes with the MCGA, while the Models 50, 60, and 80 use the VGA.

For compatibility (and, no doubt, in hopes of selling lots of hardware), IBM also offers a VGA adapter that implements the VGA subsystem on a card for the XT, AT, or PS/2 Model 30.

#### Monitors

The MCGA and VGA differ from previous IBM video adapters in that both require that you use an analog monitor instead of a digital monitor. Adapters such as the CGA and the EGA use digital monitors in which the RGB color signals generated by the adapter are digital signals (on or off). This limits the number of different colors that the subsystem can



display. For example, IBM's enhanced color display, which is driven by six RGB signals as generated by an EGA, can display a total of  $64 (2^6)$  different colors.

In contrast, the PS/2-compatible monitors use RGB signals with voltage levels that are continuously variable instead of simply on or off. Because the displayed brightness of a color corresponds to the voltage level of the color drive signals, an analog monitor can display a much larger variety of colors. directly.

From a programmer's perspective, there is not much resemblance between the MCGA and the VGA. The I/O port assignments in the MCGA and VGA differ significantly. So does the layout of *continued* 

Richard Wilton (6236 West Sixth St., Los Angeles, CA 90048) is author of The Programmer's Guide to PC and PS/2 Video Systems, which is due out in November.

Illustration: Kevin Short © 1987

IBM offers one monochrome and two color monitors for use with the MCGA and the VGA. You can use a monochrome or color monitor with either video subsystem. You can also use EGA-compatible monitors with analog capability, such as the NEC MultiSync and Sony Multi-Scan monitors, with the MCGA and VGA.

#### Compatibility

Programs that run on the CGA can run unchanged on the MCGA, even if they bypass the video BIOS and program the hardware directly. Also, because the PS/2 Model 30 has a PC-compatible bus, a monochrome display adapter or Hercules adapter can coexist with the MCGA in the PS/2 Model 30.

The VGA is similar in its programming interface to the EGA. Its control ports and buffer addressing are EGAcompatible, so programs that run on an EGA generally run on a VGA as well. The VGA is compatible enough with the EGA at the hardware level that the VGA can usually run illbehaved programs that access the EGA control registers video RAM in the two subsystems. A program that bypasses the video BIOS to control the hardware directly will probably not run on both the MCGA and the VGA unless it contains special code for programming each subsystem independently.

#### Documentation

The programming interface to PS/2 video hardware is documented in the IBM technical reference manuals for the Models 30, 50, and 60. The video BIOS is covered by a separate set of IBM reference manuals, the *Personal System/2* and *Personal Computer BIOS Interface Technical Reference*. Obviously, this article does not cover all the details of the video hardware implementation. If you need to understand the hardware or firmware in detail, you should obtain the appropriate IBM technical manuals.

#### The MCGA

The heart of the MCGA circuitry lies in two proprietary gate arrays: the memorycontroller gate array, which incorporates the functions of a CRT controller, and the video-formatter gate array, which controls video mode selection and color-attribute decoding.

You can program the memory controller through a set of 8-bit registers (see table 1) mapped to I/O ports 3D4 and 3D5 hexadecimal. [Editor's note: For the remainder of this article, addresses will be in hexadecimal.] As on the CGA, you access the registers by first writing the register number to the port at 3D4, and then writing or reading the specified register at 3D5. Unlike the CGA, however, you can read and write all the memorycontroller registers. This is a handy feature if you are debugging programs, although it's not a good idea to rely on it if you are concerned about maintaining CGA compatibility.

The first 16 memory-controller registers are analogs of the registers on the Motorola 6845, the CRT controller chip used in the CGA. This means that CGAcompatible programs that access these

 Table 1: MCGA memory-controller registers. Registers 0 through 0F

 hexadecimal are comparable to those in the CGA's CRT controller.

 Begister
 Function

| Register<br>number         | Function                                                                                                                                                                                                                                                                                                                                        |
|----------------------------|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
|                            | Horizontal total<br>Horizontal displayed<br>Start horizontal sync<br>Sync pulse width<br>Vertical total<br>Vertical total adjust<br>Vertical displayed<br>Start vertical sync<br>(Reserved)<br>Scan lines per character<br>Cursor start<br>Cursor end<br>Start address high<br>Start address low<br>Cursor location high<br>Cursor location low |
| 10<br>11<br>12<br>13<br>14 | Mode control<br>Interrupt control<br>Character generator, sync polarity<br>Character-generator pointer<br>Character-generator count                                                                                                                                                                                                             |
|                            | ~                                                                                                                                                                                                                                                                                                                                               |

Listing 1: Video mode selection using the video BIOS.

| mo <b>v</b><br>mov | ah,0<br>al,VideoModeNumber | ;AH = INT 10h function #<br>;AL= 11h (640x480 two-color)<br>; 12h (640x480 16-color) |
|--------------------|----------------------------|--------------------------------------------------------------------------------------|
| int                | 10h ;Call video BIC        | ; 13h (320x200 256-color)<br>)S                                                      |

registers directly can also run on the MCGA. Because the default horizontal and vertical CRT timing parameters used on the MCGA differ from those used on the CGA, you might want to write-protect the first seven registers so that CGA-compatible programs that attempt to update these registers do not inadvertently disrupt crucial CRT timing signals. Bit 7 of the mode-control register (register 10h) is the write-protect bit for the timing registers.

The remaining memory-controller registers control video mode selection and the alphanumeric character generator. These registers do not exist on the CGA. They support functions that are similar to what is available on the VGA: additional graphics modes and RAMloadable alphanumeric character sets.

The video formatter supports three CGA-compatible control registers. The mode-control register (I/O port 3D8) controls video mode selection. The color-control register (port 3D9) controls palette and graphics mode back-ground-color selection. The status register (port 3DA) is a read-only register whose contents indicate the status of the CRT's horizontal and vertical timing signals. All three of these registers are compatible with the analogous registers on the CGA.

In addition to the six video modes supported by the CGA, the MCGA offers a 640 by 480 two-color graphics mode (video BIOS mode 11H) and a 320 by 200 256-color graphics mode (BIOS mode 13H). You can set up both new modes, as well as all the CGA-compatible modes using INT 10h function 0 (see listing 1).

#### **New Features**

Two features of the MCGA are of special interest to programmers. One is that the vertical resolution of both alphanumeric and graphics modes is greater than on previous IBM video adapters. The other is that the MCGA can display up to 256 different colors at one time out of a possible 262,144 (256K) colors.

The vertical resolutions of the default BIOS video modes are listed in table 2. In alphanumeric modes, the vertical resolution is 400 scan lines—twice that of the CGA and better than the EGA's 350-line "enhanced" modes. Since the BIOS still displays 25 rows of characters in alphanumeric modes, the vertical size of each displayed character is 16 scan lines. These higher-resolution characters are sharp and easy to read.

When the MCGA emulates the CGA graphics modes (640 by 200 two-color and 320 by 200 four-color), it doubles the vertical size of pixels so that each is two scan lines high. Thus, although the CGA-

compatible graphics modes use the same resolution in terms of pixels, the displayed resolution is still 400 lines, so these modes have a sharper appearance on the MCGA than they do on a CGA.

Another feature of both the MCGA and the VGA is expanded color display capability. This is provided by a digital-to-analog converter (DAC) that generates the analog RGB signals used to drive the PS/2 monochrome and color monitors. (The monochrome monitor responds only to the green color signal; the color monitors recognize all three.)

The video DAC uses a set of 256 eighteen-bit internal registers, each of which specifies an RGB combination. Each of the three primary colors is allotted 6 bits of each color register; the DAC converts each 6-bit value to a corresponding analog voltage level in the signals it outputs to the monitor. Thus, the video DAC can produce any of 64 color intensities for each of the three primary colors in a color register, thereby generating 256K (64<sup>3</sup>) color combinations. Since there are 256 video DAC color registers, the video subsystem can display any 256 of the 256K color possibilities at one time.

#### Video BIOS

The video BIOS on the Model 30 provides the same set of functions as the motherboard ROM BIOS on the PC. The programming interface is the same: You access all video BIOS functions through interrupt 10h and pass parameters to the BIOS routines in the CPU's registers.

Also, several new INT 10h functions are available in the Model 30, as well as the other models in the PS/2 series (see table 3). IBM has expanded the INT 10h function 10h to provide access to the video DAC color registers. Function 12h has several new subfunctions that let you vary the default actions of other BIOS routines. For example, you can call INT 10h function 12h with the BL register set to 31h to enable or disable default palette loading when the video mode is changed.

INT 10h functions 1Ah and 1Bh are new to the PS/2 series. Your programs can call these INT 10h functions to determine the state of the video subsystem. A call to function 1Ah returns the video subsystem's display combination code, which indicates what type of monitor is in use. Function 1Bh returns a table whose contents describe the current state of the video BIOS: the current video mode, active video page, amount of video RAM available, and so on. These functions are useful in programs designed to run in more than one video mode, as well as in pop-up RAM-resident programs that must determine the current video state to produce appropriate video output.

Alphanumeric-Mode Programming Despite the MCGA's improved resolution, programming in alphanumeric modes is virtually the same as on the CGA. The important difference is that the alphanumeric character generator on the MCGA can display user-defined characters. (The EGA, VGA, Hercules Graphics Card Plus, and Hercules In-Color Card also have this capability.) The MCGA's alphanumeric character *continued* 

| Mode<br>number |                                                               | MCGA | VGA    |
|----------------|---------------------------------------------------------------|------|--------|
| 0              | 40 by 25 16-color alphanumeric<br>(320 by 400 resolution)     | х    |        |
| 0              | 40 by 25 16-color alphanumeric (360 by 400 resolution)        |      | x      |
| 2              | 80 by 25 16-color alphanumeric (640 by 400 resolution)        | х    |        |
| 2              | 80 by 25 16-color alphanumeric (720 by 400 resolution)        |      | X      |
| 11H<br>12H     | 640 by 480 two-color graphics<br>640 by 480 16-color graphics | х    | X<br>X |
| 13H            | 320 by 200 256-color graphics                                 | х    | Х      |

 Table 3: New INT 10h functions on the MCGA and VGA.

#### Function 10h: Color-palette interface

- AL=3: Toggle alphanumeric intensity/blink state
- AL=7: Read individual palette register (VGA only)
- AL=8: Read overscan (border color) register (VGA only)
- AL=9: Read all palette registers and overscan register (VGA only)
- AL=10h: Set individual video DAC color register
- AL=12h: Set block of video DAC color registers
- AL=13h: Select video DAC color page (VGA only)
- AL=15h: Read individual video DAC color register
- AL=17h: Read block of video DAC color registers
- AL=1 Ah: Read video DAC color-page state (VGA only)
- AL=1Bh: Perform gray-scale summing

#### Function 11h: Character-generator interface

- AL=4: Load 8 by 16 alphanumeric characters
- AL=14h: Set alphanumeric mode using 8 by 16 characters
- AL=24h: Load 8 by 16 graphics characters

#### Function 12h: Alternate select

- BL=30h: Select vertical resolution for alphanumeric modes (VGA only)
- BL=31h: Enable/disable default palette loading
- BL=32h: Enable/disable video addressing
- BL=33h: Enable/disable default gray-scale summing
- BL=34h: Enable/disable alphanumeric cursor emulation (VGA only)
- BL=35h: Display-switch interface

#### Function 1 Ah: Display combination code

- AL=0: Read display combination code
- AL=1: Write display combination code

#### Function 1Bh: Functionality/state information

#### Function 1Ch: Save/restore video state (VGA only)

- AL=0: Return state buffer size
- AL=1: Save video state AL=2: Restore video state



| ; load video | BIOS 8x8 characters  | ; into alphanumeric character generator |
|--------------|----------------------|-----------------------------------------|
| mov          | ax,1102h             | ; AH = INT 10h function number          |
|              |                      | ; AL = 8x8 character-set load           |
| mov          | bx,0                 | ; BX = block to load                    |
| int          | 10h                  | ; load 8x8 characters into RAM          |
| mov          | ax,1103h             | ; AH = INT 10h function number          |
|              |                      | ; AL = character-generator load         |
| mov          | bx,0                 | ; BX = blocks to load                   |
| int          | 10h                  | ; load 8x8 characters into              |
|              | ,                    | ; character generator                   |
| ; program CF | T controller to disp | lay 8x8 characters                      |
| mov          | dx,3D4h              | ; DX = MCGA I/O port address            |
| mov          | ax,309h              | ; AL = 9 (register number)              |
|              |                      | ; AH = 3 (value for register)           |
| out          | dx,ax                | ; update scan-lines register            |
| mov          | al,OAh               | ; AL = OAh (register number)            |
| out          | dx,ax                | ; update cursor-start register          |
| mov          | al,OBh               | ; AL = OBh (register number)            |
| out          | dx,ax                | ; update cursor-end register            |
| ; update sta | tus variables in vid | leo BIOS data segment                   |
| mov          | ax,40h               |                                         |
| mov          | ds,ax                | ; DS -> video BIOS data segment         |
| mov          | word ptr ds:[4Ch],   | 80*50*2 ; update CRT_LEN                |
|              |                      | 10 undete DOUE                          |
| mov          | byte ptr ds: [84h],  | 49 ; update Rows                        |



Figure 1: A video-buffer map in 640 by 480 two-color graphics mode.



Figure 2: A video-buffer map in 320 by 200 256-color graphics mode.

generator can display characters from any of four different 256-character tables defined in video RAM.

To make the MCGA's character generator display one of these character sets, you first load the bit patterns that define the characters into video RAM. Then you program the character generator to copy the bit patterns from video RAM into one of its two internal character-definition tables. (These character-definition tables are called *font pages* in the IBM technical literature.)

It is easy to use the MCGA's RAMloadable character sets on the MCGA to display characters that are smaller than the default 16-scan-line characters. In listing 2, the program calls the video BIOS to load the default graphics-mode character definitions—in which characters are only eight lines high—for use by the alphanumeric character generator. The code then reprograms the MCGA's memory controller to display only eight scan lines in each row of characters; the result is 50 rows of 80 characters each.

Apart from supporting RAM-loadable character sets, the MCGA replicates almost all the CGA's capabilities. However, the MCGA is not troubled by problems with display interference in alphanumeric modes. On the CGA, you must carefully synchronize CPU accesses to the video RAM with horizontal and vertical retrace intervals in the display refresh cycle. If you don't, you might see random patterns of interference or snow on the screen each time a program accesses the video buffer. The hardware design of the MCGA is such that this sort of display interference does not occur.

Surprisingly, the MCGA cannot generate a colored border. In alphanumeric modes on the CGA, you can display a border in any of 16 colors selected by programming the color-select register (I/O port 3D9). On the MCGA, you can still program the color-select register, but the MCGA does not display a border, regardless of the value you store in the register.

#### **Graphics-Mode Programming**

In CGA-compatible 640 by 200 twocolor and 320 by 200 four-color modes, the MCGA emulates the CGA. The system maps pixels with a two-way interleave in the video buffer at B800:0000, just as they are mapped on the CGA. Video-buffer addressing is different, however, in 640 by 480 two-color and 320 by 200 256-color modes.

The video-buffer map in both 640 by 480 two-color and 320 by 200 256-color modes starts at A000:0000. Both modes map pixels linearly in the buffer from left *continued*  to right and from top to bottom on the screen. In 640 by 480 two-color mode, a bit represents one pixel, so there are eight pixels to each byte in the video buffer and 80 bytes in the buffer per row of pixels on the screen (see figure 1).

In 320 by 200 256-color mode, each pixel value comprises 8 bits, so the video buffer is mapped as 200 320-byte rows (see figure 2).

You can write routines that manipulate pixels in these MCGA graphics modes by modifying code that runs in CGA-compatible graphics modes. The routine in listing 3 is an example of code that updates a single pixel in 320 by 200 256color mode. The program computes the video-buffer address by multiplying the number of pixels in each row by the pixel y coordinate and then adding the pixel xcoordinate. Since each byte in the buffer represents one pixel, updating a pixel consists of a single machine instruction.

The MCGA's 640 by 480 two-color graphics mode deserves attention because its horizontal resolution and vertical resolution are the same in terms of the number of pixels displayed per inch. (Programmers sometimes describe this circumstance by saying that the pixels are



#### Listing 4: Updating a video DAC color register.

| mov | ah,10h         | ;AH = INT 10h function #      |
|-----|----------------|-------------------------------|
| mov | bx,7           | ;BX = 7 (register #)          |
| mov | dh,RedValue    | ; DH,CH,CL = 6-bit RGB values |
| mov | ch, GreenValue |                               |
| mov | cl,BlueValue   |                               |
| int | 10h            | ;Call video BIOS              |
|     |                |                               |

"square.") This means that when you draw a figure in 640 by 480 graphics mode, you do not need to scale the figure to accommodate different horizontal and vertical resolutions.

#### Video DAC Programming

Programming the MCGA's video DAC is straightforward when you use the video BIOS. In all modes except the 320 by 200 256-color graphics mode, you can use only the first 16 video DAC registers. The video BIOS loads these registers by default with a set of 16 CGA-compatible color values. You can, however, update any of these color registers using any of the 256K color combinations available.

For example, listing 4 shows how you could change the color value in video DAC register 7. The color-register value is actually 18 bits in size—red, green, and blue components are each 6 bits. The higher the value you specify for each component, the higher the displayed intensity of that color.

If a monochrome display is attached to the MCGA, the video BIOS performs a gray-scaling computation before it loads a color value into the specified video DAC color register. The video BIOS performs gray-scaling by taking a weighted average of the red, green, and blue values you specify. (The formula used is 30 percent red + 59 percent green + 11 percent blue.) The result is a gray-scale value that corresponds to the overall intensity of the specified color combination.

#### The VGA

The VGA subsystem takes its name from the video graphics array, a proprietary VLSI gate-array circuit that incorporates the functions of several EGA components: the CRT controller, the sequencer, *continued* 



Figure 3: The VGA subsystem. Components of the VGA chip are outlined in red.

| Listing 5: Establishing an 80 by 50 alphanumeric mode on the VGA.      |                       |                                  |                                                                                         |   |
|------------------------------------------------------------------------|-----------------------|----------------------------------|-----------------------------------------------------------------------------------------|---|
| ; esta                                                                 | blish4                | 00-line res                      | olution in alphanumeric mode                                                            |   |
|                                                                        | mov                   | ax,1202h                         | ; AH = 12h (INT 10h function number)                                                    |   |
|                                                                        |                       |                                  | ; AL = 2 (select 400-line modes)                                                        |   |
|                                                                        | mov                   | bl,30h                           | ; BL = alphanumeric scan-lines select                                                   |   |
|                                                                        | int                   | 10h                              |                                                                                         |   |
|                                                                        | mov                   | ax,3                             | ; call video BIOS to set mode                                                           |   |
|                                                                        | int                   | 10h                              |                                                                                         |   |
| ; load video BIOS 8x8 characters into alphanumeric character generator |                       |                                  |                                                                                         |   |
|                                                                        | mov                   | ax,1112h                         | ; AH = INT 10h function number                                                          |   |
|                                                                        |                       |                                  | ; AL = 8x8 character-set load                                                           |   |
|                                                                        | mov                   | b1,0                             | ; BL = block to load                                                                    |   |
|                                                                        | int                   | 10h                              | ; load 8x8 characters into RAM                                                          |   |
| ; load                                                                 | video H<br>mov<br>mov | BIOS 8x8 cha<br>ax,1112h<br>bl,0 | ; AH = INT 10h function number<br>; AL = 8x8 character-set load<br>; BL = block to load | , |

the graphics controller, and the attribute controller (see figure 3).

You program each component of the VGA as you would on the EGA. Each contains a number of control registers mapped to 8-bit ports. As with the MCGA, you access each register by writing a register number to an I/O port and then reading or writing the specified register. The CRT controller has 25 registers addressed at ports 3D4 and 3D5; the sequencer has 5 registers at ports 3C4 and 3C5; the graphics controller maps 9 registers to 3CE and 3CF; and the attribute controller has 21 registers, including 16 palette registers, mapped to 3C0 and 3C1.

Almost all of the many VGA control registers have the same function on the EGA, so if you are familiar with the EGA, you will be comfortable programming the VGA as well. The function of each of the registers is documented in IBM's technical reference manual for the PS/2 Models 50 and 60.

The VGA supports all the video modes available on the EGA, as well as the 640 by 480 two-color and 320 by 200 256color graphics modes found on the MCGA. One additional graphics mode is unique to the VGA: a 640 by 480 16-color graphics mode (BIOS mode 12H) that is similar to the EGA-compatible 640 by 350 16-color mode, but with higher vertical resolution.

As on the MCGA, the default alphanumeric modes on the VGA have 400-line vertical resolution. Unlike the MCGA, however, you can set up the VGA's CRT controller to display alphanumeric characters with 200-line or 350-line resolution for compatibility with the CGA and the EGA.

### Video BIOS

As on the MCGA, the VGA video BIOS provides support for all the CGA- and EGA-compatible INT 10h functions. The VGA BIOS also supports INT 10h functions 1Ah and 1Bh, which return information regarding the hardware configuration and video BIOS status as they do on the MCGA.

The VGA video BIOS also provides a video-state save/restore capability through INT 10h function 1Ch. This function can save and restore all control registers, video DAC registers, and video-related information from the BIOS data area in RAM, using a buffer provided at a user-specified address. The ability to save the state of the VGA and subsequently restore it lets a program switch between video modes or program the palette and video DAC registers freely without losing the context of a previously established video state.

### **Alphanumeric-Mode Programming**

As on the MCGA, CGA-compatible alphanumeric-mode programming on the VGA is straightforward. Again, the video buffer is addressed starting at B800:0000 and mapped with alternating character codes and attributes. Video-BIOS support for character I/O is the same as it is on other IBM video subsystems.

As for the EGA and MCGA, you can configure the VGA to display user-defined alphanumeric character sets. You can also program the VGA's CRT controller to display characters of different vertical sizes, so that you can display more than the default 25 rows of alphanumeric characters. Listing 5 is a simple example of how you can call the video BIOS to set up an 80 by 50 alphanumeric mode using the 8 by 8 character definitions found in the BIOS ROM.

### **Graphics-Mode Programming**

If you can program the EGA and MCGA in graphics modes, you can program the VGA. Routines that read and write pixels continued

#### PERISCO Top-of-the-line Periscope III with real-time, hardware breakpoint board. ...Keeps you going full steam Periscope I includes a half-length board with 56K of write-protected RAM; break-out switch; ahead when other debuggers software and manual for \$345. let you down. With four Periscope II includes break-out switch; software models to pick from, you'll and manual for \$175. find a Periscope that has just Periscope II-X includes software and manual the power you need. (no hardware) for \$145. Periscope III includes a full-length board with 64K of . . Start with the model that fits your current needs. If write-protected RAM, hardware breakpoints and real-time trace buffer; you need more horsepower, upgrade for the difference in price plus \$10! break-out switch; software and manual. Periscope III for machines run-And don't worry about having a lot more to learn . . . Even when you ning up to 8 MHz is \$995; for machines running up to 10 MHz, \$1095. move to the most powerful model, Periscope III, an extra dozen commands are all that's involved. Call Toll-Free for free information or to order your **Periscope today!** Periscope's software is solid, comprehensive, and flexible. It 800-722-7006 helps you debug just about any kind of program you can write . . MAIOR CREDIT CARDS ACCEPTED. thoroughly and efficiently. Periscope's hardware adds the power to solve the really tough debugging problems. 14 BONNIE LANE ATLANTA, GA 30328 Periscope requires an IBM PC, XT, AT, or close compatible PERISCOPE (Periscope III requires hardware as well as software compatibility); DOS 404/256-3860 2.0 or later; 64K available memory; one disk drive; an 80-column Company, Inc. monitor

Circle 172 on Reader Service Card

# OUR PLUG-IN CARDS GIVE YOU PLUG-IN CONTROL.

PCXTIAT

NEW - AUTOCAD SUPPOR

Your IBM PC/XT/AT or compatible can control any IEEE-488 instrument.

You can:

Plug-in to BASIC, C, FORTRAN, or Pascal.

Use HP-IB plotters,

- printers, and instruments.
- □ Spend less time programming.

□ Call (617) 273-1818 and put us to the test.

Complete hardware and software solutions for just \$395.



Capital Equipment Corp. 99 South Bedford Street Burlington, MA. 01803

ANZANA 3.5" DRIVES AND SFIVE® SOFTWARE *I* takes More Than **DRIVE TO GET AHEA** 

Our 720K and 1.44MB internal or external 3.5" floppy drives come with software that supports more 3.5" formats on more computers than anyone else. PS/2 compatibility, Laptop compatibility, and backup capability all from the same peripheral. Please call for more information. We look forward to providing you with the finest quality 3.5" disk upgrades available.

Come see us at COMDEX/Fall '87 Bally's Las Vegas, Booth B738



Manzana MicroSystems, Inc., P.O. Box 2117, Goleta, CA 93118. Telex 4932215

## **PS/2 VIDEO PROGRAMMING**

in CGA- and MCGA-compatible graphics modes also run on the VGA in these modes. In EGA-compatible graphics modes (320 by 200 16-color, 640 by 200 16-color, and 640 by 350 16-color), the same routines you use on an EGA should also run on a VGA. The only video mode unique to the VGA is 640 by 480 16-color mode, but it is almost identical to the EGA-compatible 640 by 350 16-color mode.

In these EGA-compatible modes, the video buffer is set up as a set of four parallel bit planes, each of which shares the same range of addresses starting at A000:0000. Data bytes are transferred to and from the bit planes in parallel whenever the CPU executes a read or write instruction.

This limited parallel processing is carried out by the VGA's graphics controller, which contains a set of four 8-bit internal registers or latches. Whenever the CPU executes an instruction that performs a read from an address in the video buffer, the graphics controller copies the contents of each of the parallel bit planes at the specified address into the latches. Thus, for example, when the CPU executes a MOV reg, mem instruction, the graphics controller copies 4 bytes of data from the bit planes into the latches.

The converse process occurs when a CPU executes a write instruction. In this case, the graphics controller combines the data byte written by the CPU with the contents of each of the latches and writes the result to the bit planes. Thus, the sequence of events in updating the video buffer in graphics modes is to execute a CPU read followed by a CPU write. This can be a sequence of two MOV instructions, as well as a single CPU instruction, such as MOVS.

Pixels are represented by the set of corresponding bits at the same address in each of the bit planes. Since there are four bit planes, a pixel can have any of 16 ( $2^4$ ) different values, and the number of different colors you can display at one time is 16. You might think of the contents of the graphics controller latches as eight adjacent pixel values instead of 1 byte from each of the four bit planes.

As on the EGA, the key to graphicsmode programming on the VGA is to control the way the graphics controller manipulates the data bytes (pixel values) it reads from and writes to the bit planes. On the VGA, two graphics-controller read modes and four write modes affect what the graphics controller does during CPU reads and writes.

## **Graphics-Controller Read Modes**

The two graphics-controller read modes are the same as those implemented on the

EGA. In read mode 0, the value of one in the four latches is copied to the CPU each time the latches are loaded by a CPU read operation. In read mode 1, the eight pixel values in the latches are compared to a reference value stored in the graphics controller's color-compare register. The graphics controller returns the result of the eight comparisons in a single byte to the CPU. Each bit of the byte contains a 1 bit where a latched pixel value matches the reference value.

Read mode 0 is useful for transferring data out of the bit planes into system RAM because you can access the contents of each bit plane separately. You can use read mode 1 for graphics operations, such as region fills, where you must scan the video buffer for pixels that match a predetermined value.

#### **Graphics-Controller Write Modes**

Each of the four graphics-controller write modes is also designed to simplify certain kinds of programming tasks. Write mode 0 is the one that the video BIOS routines use most frequently. In write mode 0, the graphics controller combines the eight latched pixel values with either the data byte written by the CPU or with a pixel value stored in the graphics-controller set/reset register. The graphics controller can AND, OR, or XOR pixel values, as well as replace them with CPU or set/ reset data. You control this activity pixel by pixel by storing a bit mask in the graphics controller's bit-mask register; the bit mask indicates which of the eight latched pixel values is updated and which is left alone during the operation.

Consider what happens in listing 6, which uses write mode 0 to update the value of a pixel in 640 by 480 16-color mode. First, the routine computes the address of the pixel in the video buffer, as well as a bit-mask value for the bit-mask register. Then the graphics-controller registers are set up for the operation; write mode 0 is selected, the desired pixel value is stored in the set/reset register, the set/reset function is enabled for all four bit planes, and the bit-mask value is placed into the bit-mask register. Then the OR instruction updates the bit planes. Finally, the graphics-controller registers are updated with values that correspond to those used by default by the video BIOS, so that subsequent video BIOS routines run as expected.

Clearly, most of the work involves configuring the graphics controller; only one CPU instruction actually updates the pixel. Note the sequence of events that occurs during execution of the OR instruction: First, a CPU read occurs, so the latches are loaded with the eight pixel values at the specified address. Then, the CPU performs a logical OR of a register with the value it read from the graphics controller and performs a CPU write with the result.

The graphics controller ignores the byte written by the CPU because it is configured to use the pixel value in the set/reset register to update the latches. The bitmask-register value specifies which of the eight latched pixel values is replaced with the set/reset value as the latched data is copied to the bit planes during the CPU write operation.

In graphics-controller write mode 1, the contents of the four latches are simply copied to the bit planes. Thus, write mode 1 is useful in filling the video buffer with a solid color or a pixel pattern.

In write mode 2, the pixels in the latches are updated with the pixel value specified in the CPU data byte instead of in the set/reset register. Consequently, you can use write mode 2 as easily as write mode 0 for updating the value of individual pixels in the buffer.

The VGA also supports a graphicscontroller write mode 3. It is similar to write mode 0, except that its bit-mask value is derived by combining the data byte written by the CPU with the value in the bit-mask register using an AND operation. This lets you change the bit-mask pattern without programming the bitmask register. However, because the EGA does not support write mode 3, you must avoid using it if you are designing a program to run on the EGA as well as the VGA.

#### Video DAC Programming

Using the video DAC is somewhat more complicated on the VGA than on the MCGA because the VGA's attribute controller plays a role in accessing the video DAC. The VGA does not restrict you to using only the first 16 video DAC color *continued* 

| Listing 6: Setting the value of a pixel in 640 by 480 16-color mode. |            |                    |                                                                     |
|----------------------------------------------------------------------|------------|--------------------|---------------------------------------------------------------------|
| sp12<br>;<br>;                                                       | PROC       | near; ca           | ll with: AX = y coordinate<br>BX = x coordinate<br>CL = pixel value |
|                                                                      | te the r   | nivel addr         | ess in the video buffer                                             |
| , compu                                                              | push       | CX                 |                                                                     |
|                                                                      | mov        | cx,bx              | ; push pixei value                                                  |
|                                                                      | and        | cl,7               |                                                                     |
|                                                                      | mov        | ch,10000           | 0000b                                                               |
|                                                                      | shr        | ch,cl              | ; CH = bit mask for pixel                                           |
|                                                                      | mov        | dx,80              |                                                                     |
|                                                                      | mul        | dx                 | ; AX = Y*80                                                         |
|                                                                      | mov        | cl,3               |                                                                     |
|                                                                      | shr        | bx,cl              | ; BX = X/8                                                          |
|                                                                      | add        | bx,ax              | ; BX = Y*80 + X/8                                                   |
|                                                                      | mov        | ax,0A000           | Dh                                                                  |
|                                                                      | mov        | ds,ax              | ; DS:BX -> pixel in video buffer                                    |
| ; set u                                                              | p the gr   | aphics con         | ltroller                                                            |
|                                                                      | mov        | dx,3CEh            |                                                                     |
|                                                                      | mov        | ax,0005            |                                                                     |
|                                                                      | out        | dx,ax              | ; set up write mode O                                               |
|                                                                      | рор        | ax                 | ; pop pixel value                                                   |
|                                                                      | mov        | ah,al              |                                                                     |
|                                                                      | mov        | al,O               |                                                                     |
|                                                                      | out        | dx,ax              | ; set up set/reset register                                         |
|                                                                      | mov        | ax,OFO1h           |                                                                     |
|                                                                      |            | dx,ax              | ; set up enable set/reset register                                  |
|                                                                      |            | ah,ch              |                                                                     |
|                                                                      |            | al,8               |                                                                     |
|                                                                      | out        | dx,ax              | ; set up bit-mask register                                          |
| ; upo                                                                | late the   | -                  |                                                                     |
|                                                                      | or         | [bx],al            | ; update latches during CPU read                                    |
|                                                                      |            | 1 +                | ; update bit planes during CPU write                                |
| ; res                                                                |            |                    | phics-controller register values                                    |
|                                                                      | mov        | ax,0000            | . defeuilt act (maget velue                                         |
|                                                                      | out        | dx,ax              | ; default set/reset value                                           |
|                                                                      | mov        | ax,0001            | · default anable set (menet velue                                   |
|                                                                      | out<br>mov | dx,ax              | ; default enable set/reset value                                    |
|                                                                      | out        | ax,0FF081<br>dx,ax | n<br>; default bit-mask value                                       |
|                                                                      | ret        | αλ, αλ             | , deradro oro-mask varue                                            |
| sp12                                                                 | ENDP       |                    |                                                                     |
| obre                                                                 | LIDI       |                    |                                                                     |
|                                                                      |            |                    |                                                                     |

registers in alphanumeric modes and 16color graphics modes. You can program the attribute controller to address the 256 video DAC registers in 16 register blocks.

On the VGA, each 4-bit attribute value

(in alphanumeric modes) or 4-bit pixel value (in graphics modes) is processed by the attribute controller, which uses the value to select one of its 16 palette registers (see figure 4). Each of the palette registers contains a 6-bit value that com-



Figure 4: VGA color control.



bines with the value in the attribute controller's color-select register to form an 8-bit value; this 8-bit value is passed to the video DAC.

The video DAC in turn uses the 8-bit value to select one of its 256 color registers, each of which contains an 18-bit RGB specification. In this way, a 4-bit attribute decodes into the set of three analog RGB values output by the video subsystem to the monitor.

Both the values in the palette registers and the value in the color-select register determine which video DAC color registers are referenced to generate color output.

By default, the video BIOS maintains EGA compatibility by initializing the attribute-controller palette registers with the same 6-bit values as on the EGA, as well as the first 64 video DAC registers with RGB values that produce the same 64 colors available on the EGA.

You could use the three remaining 64register blocks of video DAC color registers by programming the attribute-controller color-select register. Video BIOS INT 10h function 10h supports this. (In IBM's technical documentation, blocks of video DAC color registers are referred to as *color pages*.)

In listing 7, I used the video BIOS to copy the contents of the first 64 video DAC registers into the second block of 64 registers. Then I called the BIOS grayscaling function to replace the second block of color-register values with their gray-scale equivalents. At this point, I could call INT 10h function 10h again to select either the default color values in the first 64 color registers or the gray-scaled values in the second 64 color registers.

#### I Could Go On...

Although there are many more PS/2 video programming techniques than I can cover in the space of this article, it is easy to draw one conclusion from this brief overview: The MCGA and the VGA fall squarely into the mainstream of IBM video subsystems. Apart from its ability to display more colors and somewhat improved resolution, the MCGA strongly resembles the CGA in its capabilities and in the way you program it. Similarly, the VGA offers nearly complete compatibility with the EGA.

The VGA represents an incremental improvement over the EGA in terms of versatility, but it does not introduce any significant improvements in speed or resolution when you compare it with the EGA or with "enhanced" EGA clones. Nevertheless, it seems that many second-source vendors of video adapters for PCs and ATs regard the VGA as a new de facto hardware standard. ■