Skip to content

maxim-zhao/bmp2tilecompressors

Repository files navigation

bmp2tile-compressors

Compression DLLs for BMP2Tile

Build status

Overview

BMP2Tile supports plug-in compressors, somewhat inspired by Winamp. This repo holds all the ones I've written, plus some that were contributed by others. As far as is possible, I try to also include decompressors for use in homebrew and ROM hacks.

Releases

Automated builds are available on the Releases page.

Compressors

DLL name Short name Longer name Description Tiles supported Tilemap supported
1bppraw 1bpp 1bpp raw (uncompressed) binary One bit per pixel tiles - discards upper bits
2bppraw 2bpp 2bpp raw (uncompressed) binary Two bits per pixel tiles - discards upper bits
3bppraw 3bpp 3bpp raw (uncompressed) binary Three bits per pixel tiles - discards upper bit
aleste aleste GG Aleste LZ Compression from the game GG Aleste
aPLib aPLib aPLib aPLib compression library
apultra apultra aPLib (apultra) apultra aPLib compressor - better compression for the same format
berlinwall berlinwallcompr Berlin Wall LZ Compression from the game The Berlin Wall
exe (configurable) (configurable) Wraps arbitrary external programs, passing data via files. This is useful if you do not want to implement your algorithm in the form of a DLL.
exomizerv3 exomizer Exomizer v3 Exomizer v3 compression
highschoolkimengumi hskcompr High School Kimengumi RLE Compression from the game High School! Kimengumi
lemmings lemmingscompr Lemmings RLE Compression from the game Lemmings
lsb lsbtilemap LSB-only tilemap Least significant byte of tilemap data
lz4 lz4 LZ4 (raw) LZ4 compression library, using smallz4. lz4ultra has negligibly different results.
lzee lzee LZEe LZEs compression library
lzsa1 lzsa1 LZSA1 LZSA compression library
lzsa2 lzsa2 LZSA2 LZSA compression library
magicknight mkre2compr Magic Knight Rayearth 2 RLE or LZ Compressor from the game 魔法騎士レイアース2 ~making of magic knight~
micromachines mmcompr Micro Machines hybrid RLE/LZ Compressor from the game Micro Machines
oapack oapack aPLib (oapack) oapack aPLib compressor - better compression for the same format
phantasystar pscompr Phantasy Star RLE Compression from the game Phantasy Star
psgaiden psgcompr PS Gaiden Compression from the game Phantasy Star Gaiden
pucrunch pucrunch Pucrunch Pucrunch algorithm
raw bin Raw (uncompressed) binary Does no compression at all
rnc1 rnc1 Rob Northen Compression type 1 Rob Northen Compression targeting better compression/slow speed
rnc2 rnc2 Rob Northen Compression type 2 Rob Northen Compression targeting faster speed/worse compression
sfg sfg Shining Force Gaiden Compression from the game Shining Force Gaiden)
shrinkler shrinkler Shrinkler Shrinkler Amiga executable compressor
sonic1 soniccompr Sonic 1 Tile compression from the game Sonic the Hedgehog
sonic2 sonic2compr Sonic 2 Tile compression from the game Sonic the Hedgehog 2
stc0 stc0compr Simple Tile Compression 0 @sverx's stc0
stc4 stc4compr Simple Tile Compression 4 @sverx's stc4
upkr upkr upkr Simple lz + arithmetic coding packer
wonderboy wbcompr Wonder Boy RLE Compression from the game Wonder Boy
wbmw wbmw Wonder Boy in Monster World RLE Compression from the game Wonder Boy in Monster World
zx0 zx0 ZX0 ZX0 compression library
zx7 zx7 ZX7 (8-bit limited) Variant of ZX7 compression library tweaked for performance

Decompressors

All size stats are for emitting data direct to VRAM on Master System, using Z80 decompressors in non-interrupt-safe mode if available. Decompression to RAM will generally use less ROM.

Description ROM (bytes) RAM (bytes, not including stack) Compression relative to "zip" Speed relative to "otir" Interrupt-safe Non-VRAM support
Aleste 90 258 64% 26% Optional
aPLib 303 5 96%
aPLib (fast) 341 0 96% 12%
Berlin Wall 241 265 74% 4%
Exomizer v3 289 768 99% 8%
High School Kimengumi (unoptimised) 119 4 67% 11%
Lemmings (unoptimised) 143 512 65% 12%
LZ4 136 0 69% 23%
LZEe 135 0 83% 17%
LZSA1 207 0 79% 25%
LZSA2 332 0 92% 18%
Magic Knight Rayearth 2 139 0 79% 22%
Micro Machines 456 0 85% 18%
Phantasy Star RLE 188 0 66% 16%
PS Gaiden 223 34 94% 14%
PS Gaiden (fast) 1028 32 94% 29%
Pucrunch (⚠ Broken) 412 44
RNC 1 1052 430 93% 3%
RNC 2 306 0 87% 17%
Shrinkler 259 2048 106% 0%
SIMS 231 2070 77% 7%
Sonic 162 8 60% 22%
Sonic 2 289 39 53% 13%
Shining Force Gaiden 269 (size of uncompressed data) 85% 25%
Simple tile Compression 0 57 0 59% 36%
Simple tile Compression 4 267 4 80% 30%
upkr 226 321 108% 1%
Wonder Boy 73 0 50% 13%
Wonder Boy in Monster World 56 0 66% 32%
ZX0 157 0 98%
ZX0 (fast) 274 0 98% 15%
ZX7 117 0 91% 14%

Note that the technologies marked with ⚠ above fail the automated benchmark tests, with crashes in the compressor or incorrect decompressed output. They could be fixed but as they are rather old, they are probably not competitive with newer compressors.

"zip" performance is compressing the data using 7z a -mx9 filename.zip. Zip has some framing overhead but it would be unfeasible to implement a decompressor on Master System. We compute the mean compression ratio of this on the test coprpus, and then compare the compressors' mean ratio, so that >100% means they compress better, and 50% means they compress to double the size on average.

Interrupt-safe decompressors can run safely while interrupts are happening and interfering with the VDP state. This will introduce some overhead - if nothing else, they need to disable interrupts - and they will therefore also change the interrupts enabled state.

Some compressors have assembly-time definitions to switch between decompressing directly to VRAM on SMS and decompressing to RAM.

Benchmark

Based on a benchmark corpus made of tilesets and title screens, here's a scatter of the results.

How to read:

  • Further to the right is faster decompression
  • Further up is better compression (smaller data)
  • 60% compression means that the compressed data is 40% of the size of the uncompressed data (for example)
  • The scatter for a particular colour group is across a corpus of realistic Master System tile data
  • Each colour group has a larger point showing the mean result across the corpus, with error bars showing one standard deviation.
  • There are lines showing the mean performance of zip and 7z on the same corpus.
  • There are two reference points to show the performance of loading uncompressed data, via the simplest and fastest methods.
  • The "Pareto front" is shown by the dotted line. This connects all the compressors which can't be beaten on compression without being slower, or can't be beaten on speed without losing some compression. This is very much dependent on my test data and averaging method, your data may have a different frontier. The fast Phantasy Star Gaiden decompressor is ignored because its lookup tables mean the code is over 1KB - you may prefer that.

There's no trivial answer to the question "which compressor is best". There can be wide variation in performance between compressors depending on your data; you may want to try a few options for your specific use case.

Don't like the graph? The underlying data is in benchmark/benchmark-results.json, I'm happy to accept improvements. If you invoke py benchmark.py plot show it will reproduce the graph.

Other compressors

Here's some other compressors people have made.

Name Link Description Tiles supported Tilemap supported
ShrunkTileMap https://github.com/sverx/STMcomp Compresses tilemaps with specific support for sequential tile indices