OS22: The VGA Rabbit Hole#

YouTube Video

If you’ve been watching for the past few days, you’d notice that we’ve been grinding on VGA. Thankfully, it’s not just me - this stuff is tough! In fact, OSDev wiki notes that, back in the day, knowing your way around VGA was enough to establish “quite a reputation” [1].

Thankfully, with the help of people who joined the stream, we were able to get to the point where we can draw on most of the screen using memory-mapped I/O, and can escape graphics mode by hitting escape without ruining the text-mode graphics.

At the end, I threw in a quick rectangle animation just for fun (which cannot be interrupted).

See issue #2 and MR !2 on GitLab for the code discussed in this post.

Motivation#

In OS15, we were able to get the screen drawing working, but unable to get back into text mode. Afterward, the code was changed to allow exiting back to text mode, but the actual VGA mode looked terrible and did not seem to support normal drawing. The goal of this post was to fix that.

How It Looked in OS15#

OS15: When VGA worked

This is what we’re going for - just a canvas that you can draw pixels on.

Here’s a link to the VGA register values used in the working implementation.

Before#

I ended up changing the main branch at some point to allow you to get back into text mode without brekaing anything when you hit ESC. Unfortunately, this completely scrambled our drawings:

Graphics Mode, Before

Thanks to Dominik Szasz on Discord for pointing this out.

After#

After much effort, we got to the point where you can switch back and forth and do some custom drawing. There’s one problem, though:

Graphics Mode, After

The vertically striped areas of the screen must be left alone, or the font will be corrupted when you switch back to text mode. Everything else can be freely drawn upon with no negative effects.

Here’s a bonus, a fun pic from along the way when some values got set incorrectly:

Graphics Mode, Bonus

Overview of VGA Registers#

There are several groups of VGA registers, each representing a separate subsystem (once a separate chip for each, but usually all-in-one for modern cards). The groups include External/General, Sequencer, Attribute Controller (AC), Graphics Controller (GC), and Cathode Ray Tube Controller (CRTC) registers. While each group has its own idiosyncracies, the technique for getting and setting the values of these registers is mostly the same across the board.

Getting and Settings Values#

VGA introduces the concept of separate registers for address and data. Since VGA has a ton of registers and I/O address space is at a premium, this approach lets you access dozens of registers using only a few I/O addresses. Here’s how it works.

  1. Get the value of the address register and save it in a variable. This is important in case you’re on an interrupt routine and someone else was using that register for something else.

  2. Write the address register with the index of the data you’d like to read.

  3. For getting the value, read the data port. For setting the value, write your value to the data port.

  4. Restore the address register by writing the value you saved from step 1 to the address port.

All of these operations use the in and out assembly instructions to interact with the CPU’s I/O address space (the I/O ports).

I/O Select Bit (Mono/Color)#

A fun little detail of some of the registers was that the I/O address changed based on whether the system was in monochrome or color mode. You can detect which mode the system is in using a bit in the External/General registers.

My solution to this was to check this bit and conditionally use either the MONO or COLOR address constant based on whether the bit was 0 or 1.

AC Register#

Another strange idiosyncracy noted in the OSDev Wiki was that the Attribute Controller’s Attribute Address Register [2], located at 0x3C0, can be used for both input and output. To get around this, you must first read port 0x3DA to ensure that 0x3C0 is in the read state, then you can use it as a normal address register. [3]

Lessons Learned#

I learned a lot of lessons from trying to make this happen. The main one is that VGA is difficult and usually doesn’t act how you’d think it would!

Not a State Machine#

First, VGA is not a state machine. Intuitively, it would make sense that if you read every single register and save the values, then the graphics card would behave the same when you re-write those values back to all the registers, right?

It turns out that this assumption is wrong. I’m not sure exactly what happens, but at some point, writing to one register affects the values of others. The order in which you write the register values seems to make a difference.

The Flavors of the Subsystems#

As I wrote the “driver” for each litle subsystem of VGA, I found that while they are all similar, each had its own “flavor.” Here’s my impression of each:

  • Graphics Controller (GC): Started here - convert existing VGA code to use our drivers

  • Attribute Controller (AC): Very similar to GC but has palette

  • External/General:

    • Had to do before CRTC b/c CRTC reg #’s dependent on this one

    • First exposure to conditional register #’s

    • Has some weird read-only function registers

  • Cathode Ray Tube Controller (CRTC): See EXT, needed an input bit from it to tell mono vs color

  • Sequencer: Short, similar to GC approach. Easy to mess everything up by accident with this one.

Text Mode Attribute Byte#

Another blatant error in the code was forgetting to set the color attribute byte when printing characters to the screen. The result was a completely black screen when returning to text mode.

It turns out that when you clear the screen in graphics mode by setting all bytes to 0, you’re setting the text-mode attribute bytes back to 0 as well. As a result, when I tried to “type” things to the screen, I was setting the text value, but the font color was black (on a black background), so nothing was visible.

The solution was to be sure to set both the character value and the color attribute byte when printing text to the screen. This change is reflected in src/screen/screen.c.

Thank you to Brigita Yantie for her questions during the stream - if she hadn’t have asked about this, I wouldn’t have found it!

VGA Font Basics#

While the details are still a mystery, I am aware of the basics of VGA fonts:

  • Each character seems to be represented by 128 bits, or 16 bytes

  • The characters are displayed on the screen as 8x16 pixels, so each 0 represents background color and each 1 represents foreground color.

  • Overwriting VGA characters in memory seems to mess everything up.

Memcpy Woes - Jon’s Address Breakthrough#

I spent a lot of time trying different ways of copying memory to back up the current state of the text-mode video memory. I have a working memcpy function that relies on the x86 movsb instruction. It seemed to work, but backing up the video memory wasn’t working as expected.

Instead, I wrote a for loop to copy each byte individually - this still did not work.

Thankfully, Jon Strong was on the stream, and he recommended that I change the address to which I was backing up the data. I initially chose what I thought was a very high address, unlikely to be used by anything else: 0x10b8000. I also tried 0x1b8000. However, it wasn’t until I used Jon’s suggestion of 0x1b800 as the backup address that things started to behave fairly normally.

The “Fix” - Beating Around the Bush#

Once the memory backup/restore was working fairly well, I started drawing on different parts of the screen and switching back to text mode to see what went wrong. Clearing the entire screen always corrupted all of the fonts, so that text mode was nothing more than a black screen. However, clearing just a quarter or half of the screen resulted in some characters being blanked out, but others remaining available.

I was able to find where the characters are mapped in VGA memory and modify them directly.

I was also able to modify all the memory around these characters with no issues when returning to text mode.

But for some reason that defies explanation, any changes I made to these font memory locations could not be undone by simply writing the old values back to that area.

So for now, you can use about 80% of the screen to draw whatever you want, but you’re stuck with a colorful stripe where the fonts live - change this, and you’ll have a hard time returning to text mode.

Next Steps#

This VGA exploration really did a number on me - I’m tired of looking into this! That being said, there are a few things left to try, keeping in mind the ultimate goal of having an easy-to-use, functional PKOS canvas that doesn’t mess up text mode.

Brute Forcing the Font Back#

Something is going wrong when modifying the addresses containing font information, but what? The only way I can think to debug is to start with one character and go from there. If I modify the 'A' character in VGA mode, but then write back the exact value for an 'A', what happens?

VESA: More modern, better docs?#

Jon Strong recommended looking into VESA, which, while less widely-supported than VGA, has more modern docs and is supposedly much easier to use.

Other Projects, eventually Filesystem#

I need to take a break from PKOS after this and work on Boom Languages CE. Once that’s up and running, it will probably make sense to plan a roadmap for PKOS. People seem to have a lot of enthusiasm for this project, so if I can break things up in a way that others can contribute, too, we could make a lot of progress in a short amount of time!

I think the main priority is being able to interact with hard drives for mass storage. I don’t even think a filesystem is necessary, just some kind of persistent storage would be a game-changer. This would open the door to storing programs and compiling them on PKOS, which could eventually lead to writing PKOS code on PKOS!

Reference#

Most of the reference material can be found in the notes folder of this blog post on GitLab. Other sources are linked directly in the relevant source code.