Difference between revisions of "UPD7220 Video Card (Revision 2)"

From Nuclear's Documentation Wiki
Jump to navigation Jump to search
m (Fix table formatting)
(Video modes tutorial; checkpoint/saving work)
Line 24: Line 24:
   
 
The uPD7220 was invented before the VGA standard, and due to this, was not designed to output a VGA signal. Since VGA was designed to drive CRT monitors, as was the uPD7220, it turns out that
 
The uPD7220 was invented before the VGA standard, and due to this, was not designed to output a VGA signal. Since VGA was designed to drive CRT monitors, as was the uPD7220, it turns out that
  +
they are fairly compatible. VGA timings work perfectly fine, but to compute them one must keep in mind the pixel and uPD7220 clocks. The uPD7220 clock is always one fourth of the card's clock
they are fairly compatible.
 
  +
speed.
  +
  +
For common VGA signal timings, I would recommend looking at this very useful site: [http://tinyvga.com/vga-timing TinyVGA signal timings]
  +
  +
One must take the timings and convert them to the correct resolution. For an example, let's convert the 640x480 VGA standard timings to a 400x400 resolution. Here are the video timings from [http://tinyvga.com/vga-timing/640x480@60Hz TinyVGA's 640x480 60hz page]:
  +
  +
{| class="wikitable"
  +
|+ Horizontal Timing
  +
|-
  +
! Region !! Pixels !! Microseconds
  +
|-
  +
| Visible area || 640 || 25.422045680238
  +
|-
  +
| Front porch || 16 || 0.63555114200596
  +
|-
  +
| Sync pulse || 96 || 3.8133068520357
  +
|-
  +
| Back porch || 48 || 1.9066534260179
  +
|-
  +
| Whole line || 800 || 31.777557100298
  +
|}
  +
  +
{| class="wikitable"
  +
|+ Vertical Timing
  +
|-
  +
! Region !! Lines !! Milliseconds
  +
|-
  +
Visible area || 480 || 15.253227408143
  +
|-
  +
Front porch || 10 || 0.31777557100298
  +
|-
  +
Sync pulse || 2 || 0.063555114200596
  +
|-
  +
Back porch || 33 || 1.0486593843098
  +
|-
  +
Whole frame || 525 || 16.683217477656
  +
|}
  +
  +
First, we need to decide on a clock frequency for our card. 24Mhz is slower than the 25.175Mhz recommended, and this will make our pixels wider by about 5%. However, a 24Mhz clock will run our uPD7220 at 6Mhz,
  +
right at the maximum frequency, and 25.175Mhz would overclock it. We will select 24Mhz, as the 5% elongation of pixels is too small to care about, and the timings should be compatible with the uPD7220 IC. We
  +
will enable wide mode, so our pixel clock is set at 24Mhz (the same as the card - without wide mode it would be 12Mhz, or half). This keeps the pixel width close to the intended width. Importantly, this change in clock speed will change the necessary timings for the video signal. Since each uPD7220 clock cycle is 166.7ns at 6Mhz, and each word is two cycles, we will be dividing the horizontal porch and sync widths
  +
by 333.3ns to get the number of words for each of the regions. We cannot compute this yet though, because the visible region still needs to be adjusted for our new resolution.
  +
  +
For our application, we need less pixels, but still need to meet the timing spec. To achieve this, we can increase the back porch and reduce the visible region until we have the resolution we want. We do this
  +
on both the x and y axes. This works because, during the back porch, the video signal is sent a black signal. On older monitors, this corresponds to a dark box around the image, and newer monitors will treat it
  +
as a blank region. Fundamentally, the monitor will still generally treat it as a 640x480, standard VGA signal, because we haven't encroached on the porches, and we haven't changed/moved the sync signals. To
  +
compute the width of the new visible area, we multiply the number of horizontal pixels by the pixel clock period: 41.67ns * 400 pixels = 16.67us. From this, we can calculate the new horizontal front porch, and
  +
then we can recompute the lines per vertical video region:
  +
  +
{| class="wikitable"
  +
|+ Updated Horizontal Timing
  +
|-
  +
! Region !! Microseconds
  +
|-
  +
| Visible area || 16.67
  +
|-
  +
| Front porch || 0.63555114200596
  +
|-
  +
| Sync pulse || 3.8133068520357
  +
|-
  +
| Back porch || 10.662032439589677
  +
|-
  +
| Whole line || 31.777557100298
  +
|}
  +
  +
The back porch becomes much wider. All we did was move the time over from the visible area to the back porch, such that the visible area has the correct number of pixels at our clock frequency. The vertical
  +
regions are adjusted similarly, but the math is much simpler, because we are only trying to keep constant the number of lines:
  +
  +
{| class="wikitable"
  +
|+ Vertical Timing
  +
|-
  +
! Region !! Lines
  +
|-
  +
Visible area || 400
  +
|-
  +
Front porch || 60
  +
|-
  +
Sync pulse || 2
  +
|-
  +
Back porch || 63
  +
|-
  +
Whole frame || 525
  +
|}
  +
  +
You will notice that I added lines to both the front and back vertical porches. This is because the uPD7220 cannot output more than 63 of each porch, so I needed to add to both to obey this limitation. For
  +
lower resolutions, it is usually acceptable to reduce the total number of lines to obey these limitations. It is important, however, that in such a case the horizontal lines are made longer to keep the refresh rate at 60hz, and that the video settings are tested to work on the monitor.
  +
  +
At this point we have the description of the mode we want, now we can convert it into the numbers used by the card. The vertical timings are already in the correct format, but the horizontal timing still must
  +
be converted.
   
 
'''Card Specific Details: Wide Mode'''
 
'''Card Specific Details: Wide Mode'''
Line 32: Line 121:
   
 
Due to wide mode support, the card's clock is 4x the uPD7220's clock. When doing calculations for video, this must be kept in mind. If wide mode is disabled, then the pixel clock is only half the card's clock. If the video timing math does not account for this, then the video signal timing will be incorrect.
 
Due to wide mode support, the card's clock is 4x the uPD7220's clock. When doing calculations for video, this must be kept in mind. If wide mode is disabled, then the pixel clock is only half the card's clock. If the video timing math does not account for this, then the video signal timing will be incorrect.
 
 
 
   
 
== Communication with the Host Processor ==
 
== Communication with the Host Processor ==

Revision as of 01:08, 14 June 2024

The uPD7220 Video Card revision 2 is a video card with 128KB of VRAM using the uPD7220 GDC or its variants. It operates at a maximum clock speed of 32Mhz. The card is based on the original uPD7220 card I (Dylan Brophy) made, which had half the VRAM and generally was far simpler. This card solves several issues with the older card, at the cost of greater size and complexity.

Detailed Specifications

  • 16-color VGA output
  • 128KB of RAM
    • 4 RAM chips, 32KB each, 25ns access and write time
    • 16 bit bus (8 bits per chip, two chips per byte)
    • 4 color bits per pixel, 4 pixels per word in RAM
      • Maximum 262144 pixels (hardware double buffering disabled)
        • Suggested resolutions: 544x480, 640x408
      • Maximum 131072 pixels (hardware double buffering enabled)
        • Suggested resolutions: 448x288, 400x300
  • uPD7220 at up to 24Mhz, Z7220 at up to 32Mhz
  • uPD7220 wide mode supported
    • Allows higher pixel clock frequencies, for higher resolutions and lower visible display times
    • Accesses two VRAM words per uPD7220 clock cycle
  • Direct access to VRAM is provided to the host CPU

Video Modes

It is assumed that the reader is familiar with VGA display timings. If not, then it is recommended to study it, or to use the video timing calculator and to leave the card in wide mode.

The uPD7220 was invented before the VGA standard, and due to this, was not designed to output a VGA signal. Since VGA was designed to drive CRT monitors, as was the uPD7220, it turns out that they are fairly compatible. VGA timings work perfectly fine, but to compute them one must keep in mind the pixel and uPD7220 clocks. The uPD7220 clock is always one fourth of the card's clock speed.

For common VGA signal timings, I would recommend looking at this very useful site: TinyVGA signal timings

One must take the timings and convert them to the correct resolution. For an example, let's convert the 640x480 VGA standard timings to a 400x400 resolution. Here are the video timings from TinyVGA's 640x480 60hz page:

Horizontal Timing
Region Pixels Microseconds
Visible area 640 25.422045680238
Front porch 16 0.63555114200596
Sync pulse 96 3.8133068520357
Back porch 48 1.9066534260179
Whole line 800 31.777557100298
Visible area || 480 || 15.253227408143 Front porch || 10 || 0.31777557100298 Sync pulse || 2 || 0.063555114200596 Back porch || 33 || 1.0486593843098 Whole frame || 525 || 16.683217477656
Vertical Timing
Region Lines Milliseconds

First, we need to decide on a clock frequency for our card. 24Mhz is slower than the 25.175Mhz recommended, and this will make our pixels wider by about 5%. However, a 24Mhz clock will run our uPD7220 at 6Mhz, right at the maximum frequency, and 25.175Mhz would overclock it. We will select 24Mhz, as the 5% elongation of pixels is too small to care about, and the timings should be compatible with the uPD7220 IC. We will enable wide mode, so our pixel clock is set at 24Mhz (the same as the card - without wide mode it would be 12Mhz, or half). This keeps the pixel width close to the intended width. Importantly, this change in clock speed will change the necessary timings for the video signal. Since each uPD7220 clock cycle is 166.7ns at 6Mhz, and each word is two cycles, we will be dividing the horizontal porch and sync widths by 333.3ns to get the number of words for each of the regions. We cannot compute this yet though, because the visible region still needs to be adjusted for our new resolution.

For our application, we need less pixels, but still need to meet the timing spec. To achieve this, we can increase the back porch and reduce the visible region until we have the resolution we want. We do this on both the x and y axes. This works because, during the back porch, the video signal is sent a black signal. On older monitors, this corresponds to a dark box around the image, and newer monitors will treat it as a blank region. Fundamentally, the monitor will still generally treat it as a 640x480, standard VGA signal, because we haven't encroached on the porches, and we haven't changed/moved the sync signals. To compute the width of the new visible area, we multiply the number of horizontal pixels by the pixel clock period: 41.67ns * 400 pixels = 16.67us. From this, we can calculate the new horizontal front porch, and then we can recompute the lines per vertical video region:

Updated Horizontal Timing
Region Microseconds
Visible area 16.67
Front porch 0.63555114200596
Sync pulse 3.8133068520357
Back porch 10.662032439589677
Whole line 31.777557100298

The back porch becomes much wider. All we did was move the time over from the visible area to the back porch, such that the visible area has the correct number of pixels at our clock frequency. The vertical regions are adjusted similarly, but the math is much simpler, because we are only trying to keep constant the number of lines:

Visible area || 400 Front porch || 60 Sync pulse || 2 Back porch || 63 Whole frame || 525
Vertical Timing
Region Lines

You will notice that I added lines to both the front and back vertical porches. This is because the uPD7220 cannot output more than 63 of each porch, so I needed to add to both to obey this limitation. For lower resolutions, it is usually acceptable to reduce the total number of lines to obey these limitations. It is important, however, that in such a case the horizontal lines are made longer to keep the refresh rate at 60hz, and that the video settings are tested to work on the monitor.

At this point we have the description of the mode we want, now we can convert it into the numbers used by the card. The vertical timings are already in the correct format, but the horizontal timing still must be converted.

Card Specific Details: Wide Mode

This card can reach quite high clock speeds for the uPD7220, and to maintain this high speed for such slow chips, wide mode is used. This allows twice as many pixels to be output per uPD7220 clock cycle. This can be used either to increase the maximum resolution, or to reduce the display time. Reducing the display time gives the uPD7220 and host processor more time to draw to the screen. On the other hand, due to the high speeds used in wide mode, there is also typically more data that must be written to the display per frame, thus putting more pressure on the host processor. The balance in display characteristics depends on the application.

Due to wide mode support, the card's clock is 4x the uPD7220's clock. When doing calculations for video, this must be kept in mind. If wide mode is disabled, then the pixel clock is only half the card's clock. If the video timing math does not account for this, then the video signal timing will be incorrect.

Communication with the Host Processor

The hardware connection uses a fairly simple 8-bit parallel bus for data exchange. Once connected, the software can read from and write to several registers and addresses on the card to control it.

Hardware Protocol

Here is a pinout of the card:

Card Pinout
Pin Name Description
1 A0 Address pin 0 (LSB)
2 A1 Address pin 1
3 A2 Address pin 2 (MSB)
4 NC No connection: this is the key pin and should be blocked
5 GND Ground connection
6 D0 Data pin 0 (LSB)
7 D1 Data pin 1
8 D2 Data pin 2
9 D3 Data pin 3
10 D4 Data pin 4
11 D5 Data pin 5
12 D6 Data pin 6
13 D7 Data pin 7 (MSB)
14 SEL Select pin (active low input) - selects the card for communication
15 5v +5V rail to power the card
16 RD Read enable pin (active low input)
17 WR Write enable pin (active low input)
18 WAIT Wait indication (active low output)


When trying to access VRAM, if the VRAM is being used by the card, then WAIT will be pulled low to tell the CPU not to end the current operation until after the WAIT pin goes high again. After WAIT goes high, wait at least 31.25ns for the read/write operation to complete before releasing any control lines or changing any data/address lines. Never read data while WAIT is low, and wait 31.25ns after WAIT goes high before reading the data to the CPU. WAIT only becomes low when trying to read from or write to VRAM; otherwise WAIT is always high.

Each access cycle should last at least 31.25ns, and SEL should be de-asserted between cycles. Any conventional processor like Z80, 8086, 6502, etc, should have no issue with WAIT or access cycles below an operating frequency of 32Mhz, regardless of the speed of the card. There may be exceptions to this, but there are no known exceptions.

Software Protocol

Card Registers
Register Purpose
0 uPD7220 command register (uPD7220 A0 = 0)
1 uPD7220 parameter register (uPD7220 A0 = 1)
2 Card configuration register (write only)
3 Direct VRAM access (accessing this may cause WAIT to go low)
4 VRAM Address bits 0-7 (write only)
5 Reserved - for now same as register 4
6 VRAM Address bits 8-15 (write only)
7 Reserved - for now same as register 6

For this card, a driver should first configure register 2 for the selected video mode and card interface configuration. This register has various bits to tell the card what to do:

Configuration Register Bits
Bit Purpose
0 Enables automatic counting of the VRAM address registers if set. If reset, then the VRAM address registers do not change when VRAM is written. More on this later.
1 Setting this bit enables the card's wide mode circutry. When reset, one pixel will be output every other clock cycle. This should match the setting on the uPD7220 IC.
2 Enables automatic double buffering. When set, draws to VRAM from the host CPU or the uPD7220 will happen on the opposite buffer as the one displaying. Setting this halfs the available VRAM for displaying.
3 Reserved
4 Reserved
5 Reserved
6 Reserved
7 Controls which buffer is displaying. Changing this bit will swap the currently displaying buffer. Only works in automatic double buffer mode.

It is recommended to keep track of what buffer is currently displaying, and perhaps this entire configuration register, so that the selected buffer can be switched. This register is important for controlling how the CPU accesses VRAM, how smooth the display is, and the way the image data is displayed.