Rendering a GBA picture

As we’ve discussed previously, GBA has these 4 objects called “windows.” Imagine it like a flashlight illuminating a wall in a dark room: the area inside the light is “inside” the window.

These 4 windows “exist”:

1) WIN0, the first window, enabled/disabled separately 2) WIN1, the second window, enabled/disabled separately 3) WINOBJ, a window defined by opaque sprite pixels drawn in mode 2, enabled and disabled separately. 4) WINOUTSIDE, literally just anything not “inside” any other window, and ALWAYS enabled if any other window is enabled

In a previous post, we discussed how to determine which window applies to any given output pixel. To determine this, we needed to know if a pixel counted as “inside” for a given window. But how do we determine that?

The way hardware does it

Hardware does this in an EXTREMELY simple manner.

In hardware, WIN0 and WIN1’s position is defined by its top, left, right, and bottom coords.

They also have a flag that works like this.

def for_each_pixel(window):
    if screen_y == window.top: window.v_flag = True
    if screen_y == window.bottom: window.v_flag = False
    
    if screen_x == window.left: window.h_flag = True
    if screen_x == window.right: window.h_flag = False
    
    window.pixel_is_inside = window.enabled and window.h_flag and window.v_flag

Note this is also done during vblank!

This extremely simple scheme is actually fairly flexible and allows for some cool effects.

What WINOBJ and WINOUTSIDE?

Determining the flag for WINOBJ depends on the particulars of your rendering setup; it has to do with if an opaque pixel was rendered for a sprite in mode 2 on that particular pixel or not.

For WINOUTSIDE, the expression is also fairly simple:

WINOUTSIDE.pixel_is_inside = (not WIN0.pixel_is_inside) and (not WIN1.pixel_is_inside) and (not WINOBJ.pixel_is_inside)

<
Previous Post
Notes on GBA PPU: How mosaic works
>
Next Post
Super Nintendo vs. Genesis/MegaDrive vs. GameBoy Advance, part 1: Intro