Compression DLLs for BMP2Tile
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.
Automated builds are available on the Releases page.
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 | ✅ | ✅ |
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.
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.
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 | ✅ |