Rendering a GBA picture

One of the visual FX available to the GBA PPU is a “mosaic” mode. But how do you implement this?

First off, a note

Horizontal mosaic is actually applied as a post-processing effect during output of the line, just before the blending stage. It’s actually fairly simple how it’s done, too!

But first let’s address vertical mosaic, as it will illuminate a few points for us.

Vertical mosaic how-to

If we think of screen rendering happening like this:

bg0.render_line(screen_y);
bg1.render_line(screen_y);
bg2.render_line(screen_y);
bg3.render_line(screen_y);
sprite.render_line(screen_y);

Vertical mosaic just changes it to this:

bg0.render_line(bg0.mosaic_enabled ? bg_mosaic_v_counter : screen_y);
bg1.render_line(bg1.mosaic_enabled ? bg_mosaic_v_counter : screen_y);
bg2.render_line(bg2.mosaic_enabled ? bg_mosaic_v_counter : screen_y);
bg3.render_line(bg3.mosaic_enabled ? bg_mosaic_v_counter : screen_y);
sprite.render_line(obj.mosaic_enabled ? obj_mosaic_v_counter : screen_y);

The line is rendered just as if it were the line held in the _mosaic_v_counter variable (with a caveat for affine backgrounds that we’ll get to).

H and V counters…how they work

For background and sprites, the GBA keeps separate H and V “mosaic counters.” These are incremented by N every N lines or columns. These counters are then used as a parameter in rendering.

So for a mosaic value of 3, the value for each row of the screen will be like this:

Row 0: 0
Row 1: 0
Row 2: 0
Row 3: 3 - n(3) rows have gone by, so increment by n(3)
Row 4: 3
Row 5: 3
Row 6: 6
Row 7: 6
Row 8: 6
Row 9: 9
and so on

The counter works exactly the same on the horizontal.

A simple, very easy way to accomplish this is like so:

if mosaic.bg.y_counter == 0:
    mosaic.bg.y_current += mosaic.bg.y_reload   # the mosaic value such as 3
    mosaic.bg.y_counter = mosaic.bg.y_reload
mosaic.bg.y_current--

Horizontal mosaic

When I first started implementing horizontal mosaic, oh my heavens was my implementation method…contrived.

It turns out the GBA just renders each line normally and applies mosaic as a sort of post-processing effect.

On the hardware, there’s a sort of a “latched pixel” for each background, that holds the value of the pixel to compare against windows, do blending on, etc.

For backgrounds, it’s very simple. This “latched pixel” is only updated with the just-rendered pixel when mosaic.bg_h_counter == 0 (so every pixel for mosaic=1, every other for mosaic=2, etc.)

That couldn’t be simpler, right?

Sprites are a TINY bit more complicated, though.

Sprite horizontal mosaic

But not a lot more complicated!

The additional complication comes in because of transparent sprite pixels, and also, sprites being per-sprite mosaic.

Our line buffer for rendering sprites (on real hardware this is rendered the line before the currently output one), contains info for each pixel:

1) If a pixel was rendered here at all, 2) What color the pixel is, 3) What priority the pixel is, 4) If the pixel is from a translucent sprite

All this is needed for a good output. However, to get good sprite mosaic output, we also need the following info:

5) If the pixel is considered mosaic

Note my phrasing there. “If the pixel is considered mosaic.” What?

On GBA hardware, sprites are rendered from 0->127, in order, onto the buffer. If a pixel already exists and we try to output a new one, we do not unless we are a lower priority, and are opaque.

HOWEVER. In order to properly blend (AND to properly do mosaic), if we are a lower priority, we ALWAYS overwrite the mosaic and priority attributes for a pixel, even if we are not an opaque pixel.

My code for this is a little fugly and looks like this:

if ((mode > 1) || (!outpx->has) || (priority < opx->priority)) { // If we are mode 2, or the output pixel is empty, or we are a higher priority...
    switch (mode) {
        case 1: // Translucent
        case 0: { // Normal
            if (c != 0) { // If we are opaque, set these attributes
                outpx->has = 1;
                outpx->color = this->ppu.palette_RAM[c + palette];
                outpx->translucent_sprite = mode;
            }
            outpx->priority = priority; // ALWAYS set these attributes
            outpx->mosaic_sprite = mosaic;
            break;
        }
        case 2: { // window sprite
            if (c != 0) w->inside[sx] = 1;
            break;
        }
    }
}

Now this is important because it preserves the ability of a sprite to produce a mosaic look even if there’s a lower-priority sprite in that particular pixel. Otherwise, we couldn’t choose properly.

Now the sprite pixel output

When choosing whether to update a background’s “pixel latch,” we simply looked at if mosaic_counter == 0. The sprite is similar but a bit more complex.

For sprites, the rule is that we update the pixel latch on a given pixel IF: 1) The currently-latched pixel does not have mosaic marked 2) The pixel we may update with, does not have mosaic marked 3) mosaic_counter == 0

if (!latched->mosaic_sprite || !src->mosaic_sprite || (mosaic_counter == 0))  {
    update_latched_pixel here
}

This allows a higher-priority mosaic sprite to properly cover up a lower-priority sprite.

Congrats on your awesome-looking mosaic effect!

One last implementation detail

On the actual hardware, mosaic counters are 4 bits, count up and are reset to 0 when they hit the mosaic #. In this case the pixel latches are still updated when the counter == 0.


<
Previous Post
Notes on GBA PPU: Windows and Blending
>
Next Post
Notes on GBA PPU: How windows work