Loading...
Searching...
No Matches
Graphics & Backgrounds Tutorial

This tutorial covers SNES graphics fundamentals including video modes, backgrounds, and tilemaps.

SNES Video Modes

The SNES has 8 video modes (0-7). OpenSNES primarily uses:

Mode BG1 BG2 BG3 BG4 Best For
0 4 colors 4 colors 4 colors 4 colors Text, simple UI
1 16 colors 16 colors 4 colors - Most games
7 256 colors (rotatable) - - - Racing, rotation effects

Setting Up Graphics

Initialize Display

#include <snes.h>
int main(void) {
// Initialize console (sets up display, enables NMI)
// Set video mode 1 (most common for games)
// Enable BG1 on main screen
// Turn on display
while (1) {
}
return 0;
}
int main(void)
Entry point — initialize audio, display controls, run transport loop.
Definition main.c:37
void consoleInit(void)
Initialize SNES hardware.
void WaitForVBlank(void)
Wait for next VBlank period.
void setScreenOn(void)
Enable screen display.
#define REG_TM
Main screen designation (W)
Definition registers.h:181
#define TM_BG1
Definition registers.h:439
#define BGMODE_MODE1
Definition registers.h:429
OpenSNES Master Header.
void setMode(u8 mode, u8 flags)
Set background mode.

Loading Tiles to VRAM

Tiles are 8x8 pixel graphics stored in VRAM. For 4bpp (16 color) tiles:

// Tile data (32 bytes per 8x8 tile for 4bpp)
const u8 my_tiles[] = {
// Plane 0-1 (16 bytes)
0xFF, 0x00, 0xFF, 0x00, 0xFF, 0x00, 0xFF, 0x00,
0xFF, 0x00, 0xFF, 0x00, 0xFF, 0x00, 0xFF, 0x00,
// Plane 2-3 (16 bytes)
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00
};
void load_tiles(void) {
u16 i;
// Set VRAM address (tiles at $1000)
REG_VMAIN = 0x80; // Increment on high byte
REG_VMADDL = 0x00;
REG_VMADDH = 0x10; // $1000
// Copy tile data
for (i = 0; i < sizeof(my_tiles); i += 2) {
}
}
static u8 i
Definition main.c:156
static u16 bx
Definition main.c:159
#define REG_VMADDH
VRAM address high (W)
Definition registers.h:118
#define REG_VMAIN
VRAM address increment mode (W)
Definition registers.h:112
#define REG_VMADDL
VRAM address low (W)
Definition registers.h:115
#define REG_VMDATAL
VRAM data write low (W)
Definition registers.h:121
#define REG_VMDATAH
VRAM data write high (W)
Definition registers.h:124
unsigned short u16
16-bit unsigned integer (0 to 65535)
Definition types.h:52
unsigned char u8
8-bit unsigned integer (0 to 255)
Definition types.h:46

Setting Up Tilemaps

Tilemaps define which tiles appear where on screen. Each entry is 2 bytes:

Bits: vhopppcc cccccccc
v = vertical flip
h = horizontal flip
o = priority
ppp = palette (0-7)
cccccccccc = tile number (0-1023)
void setup_tilemap(void) {
u16 x, y;
// Set VRAM address for tilemap ($0400)
REG_VMADDL = 0x00;
REG_VMADDH = 0x04;
// Fill 32x32 tilemap
for (y = 0; y < 32; y++) {
for (x = 0; x < 32; x++) {
REG_VMDATAL = 0x01; // Tile number 1
REG_VMDATAH = 0x00; // No flip, palette 0
}
}
}

Setting Palettes

SNES uses 15-bit color (RGB555):

void setup_palette(void) {
// Set CGRAM address (palette 0)
REG_CGADD = 0;
// Color 0: Black (transparent for BG)
REG_CGDATA = 0x00;
REG_CGDATA = 0x00;
// Color 1: White
REG_CGDATA = 0xFF;
REG_CGDATA = 0x7F;
// Color 2: Red
REG_CGDATA = 0x1F;
REG_CGDATA = 0x00;
// Color 3: Green
REG_CGDATA = 0xE0;
REG_CGDATA = 0x03;
// Color 4: Blue
REG_CGDATA = 0x00;
REG_CGDATA = 0x7C;
}
#define REG_CGADD
CGRAM address (W)
Definition registers.h:148
#define REG_CGDATA
CGRAM data write (W)
Definition registers.h:151

Background Scrolling

void update_scroll(void) {
// Write scroll registers (write twice for 16-bit value)
REG_BG1HOFS = (u8)(scroll_x >> 8);
REG_BG1VOFS = (u8)(scroll_y >> 8);
}
#define REG_BG1VOFS
BG1 vertical scroll (W, 2x write)
Definition registers.h:91
#define REG_BG1HOFS
BG1 horizontal scroll (W, 2x write)
Definition registers.h:88
signed short s16
16-bit signed integer (-32768 to 32767)
Definition types.h:49

Parallax Scrolling

For depth effect, scroll backgrounds at different speeds:

// In main loop:
// BG1 (foreground) - full speed
// BG2 (background) - half speed
REG_BG2HOFS = (u8)((scroll_x >> 1) >> 8);
#define REG_BG2HOFS
BG2 horizontal scroll (W, 2x write)
Definition registers.h:94

Using gfx4snes

Convert PNG images to SNES format using gfx4snes (based on PVSnesLib):

# Convert 8x8 tiles with palette output
gfx4snes -s 8 -p -i tiles.png
# Output: tiles.pic (tile data), tiles.pal (palette)
# Convert 16x16 sprites with palette
gfx4snes -s 16 -p -i sprites.png
# Convert with metasprite definition (for animation)
gfx4snes -s 16 -p -T -X 32 -Y 48 -i character.png
# Output: character.pic, character.pal, character_meta.inc

Include the generated data in your assembly file:

sprite_tiles:
.incbin "res/sprites.pic"
sprite_tiles_end:
sprite_pal:
.incbin "res/sprites.pal"
sprite_pal_end:

Example: Complete Background Setup

See examples/graphics/effects/parallax_scrolling/ for a complete parallax scrolling example.

Next Steps