Update to rev5. Implement (de)compression
This commit is contained in:
parent
e08b2c74b9
commit
bf81221ab1
7 changed files with 170 additions and 83 deletions
3
.gitmodules
vendored
Normal file
3
.gitmodules
vendored
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
[submodule "lzsa"]
|
||||||
|
path = lzsa
|
||||||
|
url = https://github.com/emmanuel-marty/lzsa.git
|
144
bitmapx16.cpp
144
bitmapx16.cpp
|
@ -1,4 +1,5 @@
|
||||||
#include "bitmapx16.h"
|
#include "bitmapx16.h"
|
||||||
|
#include <lib.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <fstream>
|
#include <fstream>
|
||||||
#include <filesystem>
|
#include <filesystem>
|
||||||
|
@ -16,9 +17,6 @@ float BitmapX16::closeness_to_color(PaletteEntry a, PaletteEntry b) {
|
||||||
void BitmapX16::set_bpp(uint8_t bpp) {
|
void BitmapX16::set_bpp(uint8_t bpp) {
|
||||||
this->bpp = bpp;
|
this->bpp = bpp;
|
||||||
quantize_colors = true;
|
quantize_colors = true;
|
||||||
if (get_significant() >= (1 << bpp)) {
|
|
||||||
set_significant((1 << bpp) - 1);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
uint8_t BitmapX16::get_bpp() const {
|
uint8_t BitmapX16::get_bpp() const {
|
||||||
return bpp;
|
return bpp;
|
||||||
|
@ -81,6 +79,8 @@ void BitmapX16::write_x16(const char *filename) {
|
||||||
size_t bufsize;
|
size_t bufsize;
|
||||||
uint8_t pixels_per_byte;
|
uint8_t pixels_per_byte;
|
||||||
vector<uint8_t> pixels;
|
vector<uint8_t> pixels;
|
||||||
|
vector<uint8_t> outpixels;
|
||||||
|
size_t outpixelsize;
|
||||||
size_t pixelCount;
|
size_t pixelCount;
|
||||||
pixels_per_byte = (8/bpp);
|
pixels_per_byte = (8/bpp);
|
||||||
apply();
|
apply();
|
||||||
|
@ -89,8 +89,10 @@ void BitmapX16::write_x16(const char *filename) {
|
||||||
printf("Image size: (%lu, %lu)\n", w, h);
|
printf("Image size: (%lu, %lu)\n", w, h);
|
||||||
pixelCount = w * h * 3;
|
pixelCount = w * h * 3;
|
||||||
pixels.resize(pixelCount);
|
pixels.resize(pixelCount);
|
||||||
bufsize = 512+32+((w*h)/pixels_per_byte);
|
bufsize = palette_entries.size()*2+32;
|
||||||
buf.resize(bufsize);
|
buf.resize(bufsize);
|
||||||
|
outpixelsize = ((w*h)/pixels_per_byte);
|
||||||
|
outpixels.resize(outpixelsize);
|
||||||
memset(buf.data(), 0, bufsize);
|
memset(buf.data(), 0, bufsize);
|
||||||
buf[0] = 0x42;
|
buf[0] = 0x42;
|
||||||
buf[1] = 0x4D;
|
buf[1] = 0x4D;
|
||||||
|
@ -120,14 +122,19 @@ void BitmapX16::write_x16(const char *filename) {
|
||||||
buf[7] = w >> 8;
|
buf[7] = w >> 8;
|
||||||
buf[8] = h;
|
buf[8] = h;
|
||||||
buf[9] = h >> 8;
|
buf[9] = h >> 8;
|
||||||
buf[10] = extra_to_real_palette(border);
|
buf[10] = palette_entries.size();
|
||||||
buf[11] = significant_count;
|
buf[11] = significant_start;
|
||||||
buf[12] = significant_start;
|
uint16_t image_start = 32+(2*palette_entries.size())+1;
|
||||||
for (size_t i = 13; i < 32; i++) {
|
buf[12] = image_start;
|
||||||
|
buf[13] = image_start>>8;
|
||||||
|
buf[14] = compress ? 255 : 0;
|
||||||
|
--image_start;
|
||||||
|
for (size_t i = 15; i < 31; i++) {
|
||||||
buf[i] = 0; // Reserved bytes.
|
buf[i] = 0; // Reserved bytes.
|
||||||
}
|
}
|
||||||
for (size_t i = 0; i < 256; i++) {
|
buf[31] = extra_to_real_palette(border);
|
||||||
palette[i].write(buf.data() + (32+(i*2)));
|
for (size_t i = 0; i < palette_entries.size(); i++) {
|
||||||
|
palette_entries[i].write(buf.data() + (32+(i*2)));
|
||||||
}
|
}
|
||||||
for (size_t i = 0, x = 0, y = 0; i < (w * h); i++, x++) {
|
for (size_t i = 0, x = 0, y = 0; i < (w * h); i++, x++) {
|
||||||
if (x >= w) {
|
if (x >= w) {
|
||||||
|
@ -135,11 +142,23 @@ void BitmapX16::write_x16(const char *filename) {
|
||||||
y += 1;
|
y += 1;
|
||||||
}
|
}
|
||||||
ColorRGB px = image->pixelColor(x, y);
|
ColorRGB px = image->pixelColor(x, y);
|
||||||
size_t imagestart = (32+512);
|
|
||||||
size_t pixelIdx = get_pixel_idx(x, y);
|
size_t pixelIdx = get_pixel_idx(x, y);
|
||||||
size_t imagebyteidx = get_byte_idx(pixelIdx);
|
size_t imagebyteidx = get_byte_idx(pixelIdx);
|
||||||
uint8_t pixelinbyte = get_inner_idx(pixelIdx);
|
uint8_t pixelinbyte = get_inner_idx(pixelIdx);
|
||||||
buf[imagestart + imagebyteidx] |= (color_to_palette_entry(px) & get_bitmask()) << (bpp * pixelinbyte);
|
outpixels[imagebyteidx] |= (color_to_palette_entry(px) & get_bitmask()) << (bpp * pixelinbyte);
|
||||||
|
}
|
||||||
|
bufsize += outpixelsize;
|
||||||
|
buf.resize(bufsize);
|
||||||
|
if (compress) {
|
||||||
|
size_t compressed_size = lzsa_compress_inmem(outpixels.data(), buf.data() + image_start, outpixelsize, bufsize - image_start, LZSA_FLAG_RAW_BLOCK, 1, 2);
|
||||||
|
if (compressed_size == (size_t)-1) {
|
||||||
|
printf("Error compressing file\n");
|
||||||
|
throw std::exception();
|
||||||
|
}
|
||||||
|
bufsize -= outpixelsize - compressed_size;
|
||||||
|
buf.resize(bufsize);
|
||||||
|
} else {
|
||||||
|
memcpy(buf.data() + image_start, outpixels.data(), outpixelsize);
|
||||||
}
|
}
|
||||||
printf("Writing output file %s...\n", filename);
|
printf("Writing output file %s...\n", filename);
|
||||||
std::ofstream outfile(filename,std::ofstream::binary);
|
std::ofstream outfile(filename,std::ofstream::binary);
|
||||||
|
@ -151,7 +170,11 @@ void BitmapX16::load_x16(const char *filename) {
|
||||||
vector<uint8_t> buf;
|
vector<uint8_t> buf;
|
||||||
size_t bufsize = 0;
|
size_t bufsize = 0;
|
||||||
size_t bufpos = 0;
|
size_t bufpos = 0;
|
||||||
|
uint8_t palette_used = 0;
|
||||||
uint8_t pixels_per_byte;
|
uint8_t pixels_per_byte;
|
||||||
|
uint16_t image_start = 0;
|
||||||
|
bool compressed = false;
|
||||||
|
vector<uint8_t> decompression_buffer;
|
||||||
vector<uint8_t> pixels;
|
vector<uint8_t> pixels;
|
||||||
bufsize = 3;
|
bufsize = 3;
|
||||||
buf.resize(bufsize);
|
buf.resize(bufsize);
|
||||||
|
@ -173,7 +196,7 @@ void BitmapX16::load_x16(const char *filename) {
|
||||||
throw std::exception();
|
throw std::exception();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
bufsize += 10;
|
bufsize += 12;
|
||||||
buf.resize(bufsize);
|
buf.resize(bufsize);
|
||||||
infile.read((char*)buf.data() + bufpos, bufsize - bufpos);
|
infile.read((char*)buf.data() + bufpos, bufsize - bufpos);
|
||||||
bufpos += bufsize - bufpos;
|
bufpos += bufsize - bufpos;
|
||||||
|
@ -187,24 +210,37 @@ void BitmapX16::load_x16(const char *filename) {
|
||||||
w = buf[6] | (buf[7] << 8);
|
w = buf[6] | (buf[7] << 8);
|
||||||
h = buf[8] | (buf[9] << 8);
|
h = buf[8] | (buf[9] << 8);
|
||||||
printf("Image size: (%lu, %lu)\n", w, h);
|
printf("Image size: (%lu, %lu)\n", w, h);
|
||||||
border = buf[10];
|
palette_used = buf[10];
|
||||||
significant_count = buf[11];
|
significant_start = buf[11];
|
||||||
significant_start = buf[12];
|
significant_count = palette_used;
|
||||||
bufsize += 19;
|
image_palette_count = 0;
|
||||||
bufsize += 512;
|
image_start = buf[12] | (buf[13] << 8);
|
||||||
bufsize += (w * h)/pixels_per_byte;
|
if ((int8_t)buf[14] == -1) {
|
||||||
|
compressed = true;
|
||||||
|
}
|
||||||
|
--image_start;
|
||||||
|
bufsize = std::filesystem::file_size(filename);
|
||||||
buf.resize(bufsize);
|
buf.resize(bufsize);
|
||||||
infile.read((char*)buf.data() + bufpos, bufsize - bufpos);
|
infile.read((char*)buf.data() + bufpos, bufsize - bufpos);
|
||||||
bufpos += bufsize - bufpos;
|
bufpos += bufsize - bufpos;
|
||||||
|
border = buf[31];
|
||||||
for (size_t i = 0; i < 256; i++) {
|
palette_entries.clear();
|
||||||
palette[(uint8_t)(i-significant_start)] = PaletteEntry(buf.data() + (32+(i*2)));
|
for (size_t i = 0; i < palette_used; i++) {
|
||||||
|
palette_entries.push_back(PaletteEntry(buf.data() + (32+(i*2))));
|
||||||
}
|
}
|
||||||
// Border is always an extra palette entry.
|
|
||||||
extra_palette_entries.push_back(palette[border]);
|
|
||||||
border = extra_palette_entries.size() - 1;
|
|
||||||
// Get pixel vector for later use as image data.
|
// Get pixel vector for later use as image data.
|
||||||
pixels.resize(w * h * 3);
|
pixels.resize(w * h * 3);
|
||||||
|
decompression_buffer.resize(w*h/pixels_per_byte);
|
||||||
|
if (compressed) {
|
||||||
|
int version;
|
||||||
|
size_t bytes = lzsa_decompress_inmem(buf.data() + image_start, decompression_buffer.data(), bufsize - image_start, decompression_buffer.size(), LZSA_FLAG_RAW_BLOCK, &version);
|
||||||
|
if (bytes == (size_t)-1) {
|
||||||
|
printf("Error decompressing file!\n");
|
||||||
|
throw std::exception();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
memcpy(decompression_buffer.data(), buf.data() + image_start, decompression_buffer.size());
|
||||||
|
}
|
||||||
size_t outpixelidx = 0;
|
size_t outpixelidx = 0;
|
||||||
for (size_t i = 0, x = 0, y = 0; i < (w * h); i++, x++) {
|
for (size_t i = 0, x = 0, y = 0; i < (w * h); i++, x++) {
|
||||||
// Make sure Y is incremented when necessary.
|
// Make sure Y is incremented when necessary.
|
||||||
|
@ -213,17 +249,19 @@ void BitmapX16::load_x16(const char *filename) {
|
||||||
y += 1;
|
y += 1;
|
||||||
}
|
}
|
||||||
// Get the required data.
|
// Get the required data.
|
||||||
size_t imagestart = (32+512);
|
|
||||||
size_t pixelIdx = get_pixel_idx(x, y);
|
size_t pixelIdx = get_pixel_idx(x, y);
|
||||||
size_t imagebyteidx = get_byte_idx(pixelIdx);
|
size_t imagebyteidx = get_byte_idx(pixelIdx);
|
||||||
uint8_t pixelinbyte = get_inner_idx(pixelIdx);
|
uint8_t pixelinbyte = get_inner_idx(pixelIdx);
|
||||||
uint8_t paletteidx = (buf[imagestart + imagebyteidx] >> (pixelinbyte * bpp)) & (get_bitmask());
|
uint8_t paletteidx = (decompression_buffer[imagebyteidx] >> (pixelinbyte * bpp)) & (get_bitmask());
|
||||||
PaletteEntry entry = palette[paletteidx];
|
PaletteEntry entry = palette_entries[paletteidx];
|
||||||
uint8_t r = entry.r << 4, g = entry.g << 4, b = entry.b << 4;
|
uint8_t r = entry.r << 4, g = entry.g << 4, b = entry.b << 4;
|
||||||
// Add the pixel data to the pixels array.
|
// Add the pixel data to the pixels array.
|
||||||
pixels[outpixelidx++] = r;
|
pixels[outpixelidx++] = r;
|
||||||
pixels[outpixelidx++] = g;
|
pixels[outpixelidx++] = g;
|
||||||
pixels[outpixelidx++] = b;
|
pixels[outpixelidx++] = b;
|
||||||
|
if (paletteidx > image_palette_count+significant_start) {
|
||||||
|
image_palette_count = paletteidx-significant_start;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
// Create the Magick++ image
|
// Create the Magick++ image
|
||||||
image = new Image(w, h, "RGB", CharPixel, pixels.data());
|
image = new Image(w, h, "RGB", CharPixel, pixels.data());
|
||||||
|
@ -238,13 +276,6 @@ void BitmapX16::write_pc(const char *filename) {
|
||||||
}
|
}
|
||||||
image->write(filename);
|
image->write(filename);
|
||||||
}
|
}
|
||||||
PaletteEntry BitmapX16::get_palette_entry(uint8_t idx, bool extra) const {
|
|
||||||
if (extra) {
|
|
||||||
return extra_palette_entries.at(idx);
|
|
||||||
} else {
|
|
||||||
return palette[idx];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
void BitmapX16::load_pc(const char *filename) {
|
void BitmapX16::load_pc(const char *filename) {
|
||||||
image = new Image(filename);
|
image = new Image(filename);
|
||||||
w = image->columns();
|
w = image->columns();
|
||||||
|
@ -274,7 +305,7 @@ uint8_t BitmapX16::get_orable_pixel(uint8_t pixelinbyte, uint8_t color) {
|
||||||
}
|
}
|
||||||
|
|
||||||
BitmapX16::BitmapX16() {
|
BitmapX16::BitmapX16() {
|
||||||
extra_palette_entries = vector<PaletteEntry>();
|
palette_entries = vector<PaletteEntry>();
|
||||||
}
|
}
|
||||||
void BitmapX16::generate_palette() {
|
void BitmapX16::generate_palette() {
|
||||||
uint16_t max = (uint16_t)image->colorMapSize();
|
uint16_t max = (uint16_t)image->colorMapSize();
|
||||||
|
@ -296,27 +327,22 @@ void BitmapX16::generate_palette() {
|
||||||
significant_count = max;
|
significant_count = max;
|
||||||
}
|
}
|
||||||
bitmask = (1 << bpp) - 1;
|
bitmask = (1 << bpp) - 1;
|
||||||
for (uint16_t i = 0; i < significant_start; i++) {
|
|
||||||
palette[i] = PaletteEntry();
|
|
||||||
}
|
|
||||||
for (uint16_t i = significant_start; i < max+significant_start; i++) {
|
|
||||||
ColorRGB map_color = image->colorMap(i-significant_start);
|
|
||||||
palette[i] = PaletteEntry(map_color);
|
|
||||||
}
|
|
||||||
image_palette_count = max;
|
image_palette_count = max;
|
||||||
for (uint16_t i = max+significant_start; i < 256; i++) {
|
palette_entries.clear();
|
||||||
if ((uint16_t)extra_palette_entries.size() > i - (max+significant_start)) {
|
for (uint16_t i = 0; i < image_palette_count; i++) {
|
||||||
palette[i] = extra_palette_entries[i - (max+significant_start)];
|
ColorRGB map_color = image->colorMap(i);
|
||||||
} else {
|
palette_entries.push_back(PaletteEntry(map_color));
|
||||||
palette[i] = PaletteEntry();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
for (uint16_t i = 0; i < extra_palette_entries.size(); i++) {
|
||||||
|
palette_entries.push_back(extra_palette_entries[i]);
|
||||||
|
}
|
||||||
|
significant_count = palette_entries.size();
|
||||||
if (debug & DebugShowPalette) {
|
if (debug & DebugShowPalette) {
|
||||||
for (size_t i = 0; i < 256; i++) {
|
for (size_t i = 0; i < palette_entries.size(); i++) {
|
||||||
uint8_t significant_end = significant_start+significant_count;
|
uint8_t significant_end = significant_start+image_palette_count;
|
||||||
bool significant = i >= significant_start && i < significant_end;
|
bool significant = i >= significant_start && i < significant_end;
|
||||||
bool extra = i >= significant_end && i < significant_end+extra_palette_entries.size();
|
bool extra = i >= image_palette_count && i < image_palette_count+extra_palette_entries.size();
|
||||||
printf("palette[%02x] = %s %s\n", (uint16_t)i, palette[i].to_string().c_str(), significant ? "(Significant)" : extra ? "(Extra)" : "");
|
printf("palette[%02x] = %s %s\n", (uint16_t)i, palette_entries[i].to_string().c_str(), significant ? "(Significant)" : extra ? "(Extra)" : "");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -331,8 +357,8 @@ uint8_t BitmapX16::color_to_palette_entry(const ColorRGB &rgb) {
|
||||||
if (debug & DebugShowCloseness) {
|
if (debug & DebugShowCloseness) {
|
||||||
printf("Closest color for %s: ", color.to_string().c_str());
|
printf("Closest color for %s: ", color.to_string().c_str());
|
||||||
}
|
}
|
||||||
for (size_t i = significant_start; i < significant_count+significant_start; i++) {
|
for (size_t i = 0; i < image_palette_count; i++) {
|
||||||
float possibility_closeness = closeness_to_color(palette[i], color);
|
float possibility_closeness = closeness_to_color(palette_entries[i], color);
|
||||||
//printf("Closeness: %f", possibility_closeness);
|
//printf("Closeness: %f", possibility_closeness);
|
||||||
if (possibility_closeness < closeness) {
|
if (possibility_closeness < closeness) {
|
||||||
output = i;
|
output = i;
|
||||||
|
@ -340,7 +366,7 @@ uint8_t BitmapX16::color_to_palette_entry(const ColorRGB &rgb) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (debug & DebugShowCloseness) {
|
if (debug & DebugShowCloseness) {
|
||||||
PaletteEntry output_entry = palette[output];
|
PaletteEntry output_entry = palette_entries[output];
|
||||||
printf("%s\n", output_entry.to_string().c_str());
|
printf("%s\n", output_entry.to_string().c_str());
|
||||||
}
|
}
|
||||||
//PaletteEntry entry = palette[output];
|
//PaletteEntry entry = palette[output];
|
||||||
|
@ -370,3 +396,9 @@ uint8_t BitmapX16::get_border_color() const {
|
||||||
void BitmapX16::set_border_color(uint8_t idx) {
|
void BitmapX16::set_border_color(uint8_t idx) {
|
||||||
border = idx;
|
border = idx;
|
||||||
}
|
}
|
||||||
|
void BitmapX16::enable_compression(bool enabled) {
|
||||||
|
compress = enabled;
|
||||||
|
}
|
||||||
|
bool BitmapX16::compression_enabled() const {
|
||||||
|
return compress;
|
||||||
|
}
|
19
bitmapx16.h
19
bitmapx16.h
|
@ -12,9 +12,9 @@ enum BitmapX16DebugFlags : int {
|
||||||
class BitmapX16 {
|
class BitmapX16 {
|
||||||
/// \brief Bits per pixel of the image
|
/// \brief Bits per pixel of the image
|
||||||
uint8_t bpp = 0;
|
uint8_t bpp = 0;
|
||||||
/// \brief The palette entries used within this image
|
/// \brief The list of palette entries.
|
||||||
PaletteEntry palette[256];
|
vector<PaletteEntry> palette_entries;
|
||||||
/// \brief Any extra palette entries to add.
|
/// \brief A copy of the extra palette entries.
|
||||||
vector<PaletteEntry> extra_palette_entries;
|
vector<PaletteEntry> extra_palette_entries;
|
||||||
/// \brief The amount of colors used within the palette
|
/// \brief The amount of colors used within the palette
|
||||||
uint8_t significant_count = 0;
|
uint8_t significant_count = 0;
|
||||||
|
@ -24,6 +24,8 @@ class BitmapX16 {
|
||||||
Image *image = nullptr;
|
Image *image = nullptr;
|
||||||
/// \brief Set to true to queue color quantization. Set automatically when needed.
|
/// \brief Set to true to queue color quantization. Set automatically when needed.
|
||||||
bool quantize_colors = false;
|
bool quantize_colors = false;
|
||||||
|
/// \brief Enables LZSA compression
|
||||||
|
bool compress = false;
|
||||||
/// \brief Current width
|
/// \brief Current width
|
||||||
size_t w = 0;
|
size_t w = 0;
|
||||||
/// \brief Current height
|
/// \brief Current height
|
||||||
|
@ -88,11 +90,6 @@ class BitmapX16 {
|
||||||
/// \param entry The new entry to add
|
/// \param entry The new entry to add
|
||||||
/// \returns The palette index within the extra palette entries of the new entry
|
/// \returns The palette index within the extra palette entries of the new entry
|
||||||
uint8_t add_palette_entry(PaletteEntry entry);
|
uint8_t add_palette_entry(PaletteEntry entry);
|
||||||
/// \brief Obtains a palette entry within the image
|
|
||||||
/// \param id The index of the palette entry
|
|
||||||
/// \param extra Set to true for an extra palette entry, false for a normal palette entry.
|
|
||||||
/// \returns The palette entry requested.
|
|
||||||
PaletteEntry get_palette_entry(uint8_t id, bool extra) const;
|
|
||||||
/// \brief Sets the bits per pixel of the image
|
/// \brief Sets the bits per pixel of the image
|
||||||
/// \param bpp The bits per pixel of the image, one of 1, 2, 4, or 8
|
/// \param bpp The bits per pixel of the image, one of 1, 2, 4, or 8
|
||||||
void set_bpp(uint8_t bpp);
|
void set_bpp(uint8_t bpp);
|
||||||
|
@ -124,6 +121,12 @@ class BitmapX16 {
|
||||||
/// \brief Returns the status of the dithering flag
|
/// \brief Returns the status of the dithering flag
|
||||||
/// \returns The value of the dithering flag
|
/// \returns The value of the dithering flag
|
||||||
bool dithering_enabled() const;
|
bool dithering_enabled() const;
|
||||||
|
/// \brief Enables or disables LZSA compression
|
||||||
|
/// \param enabled Pass true to enable, false to disable
|
||||||
|
void enable_compression(bool enabled);
|
||||||
|
/// \brief Returns the status of the LZSA compression flag
|
||||||
|
/// \returns The value of the compression flag.
|
||||||
|
bool compression_enabled() const;
|
||||||
/// \brief Applies queued operations to the internal representation of the image
|
/// \brief Applies queued operations to the internal representation of the image
|
||||||
void apply();
|
void apply();
|
||||||
/// \brief Applies queued operations and writes the image to a PC-compatible file
|
/// \brief Applies queued operations and writes the image to a PC-compatible file
|
||||||
|
|
1
lzsa
Submodule
1
lzsa
Submodule
|
@ -0,0 +1 @@
|
||||||
|
Subproject commit 15ee2dfe118eeb8f7683ca44f64821c3a61ca1e5
|
12
main.cpp
12
main.cpp
|
@ -7,6 +7,7 @@
|
||||||
#include <vector>
|
#include <vector>
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <exception>
|
#include <exception>
|
||||||
|
#include <lib.h>
|
||||||
#include "bitmapx16.h"
|
#include "bitmapx16.h"
|
||||||
using std::vector;
|
using std::vector;
|
||||||
using std::map;
|
using std::map;
|
||||||
|
@ -35,6 +36,8 @@ void usage() {
|
||||||
printf("\tEnables debugging flags. May contain any number of single-letter debugging flags.\n");
|
printf("\tEnables debugging flags. May contain any number of single-letter debugging flags.\n");
|
||||||
printf("\tFlags: p = show palette entry, c = show palette closeness lookups\n");
|
printf("\tFlags: p = show palette entry, c = show palette closeness lookups\n");
|
||||||
printf("\tUse an ! before a flag to disable it.\n");
|
printf("\tUse an ! before a flag to disable it.\n");
|
||||||
|
printf("-compress\n");
|
||||||
|
printf("\tCompresses the image with LZSA compression\n");
|
||||||
printf("-help\n");
|
printf("-help\n");
|
||||||
printf("\tDisplays this help message.\n");
|
printf("\tDisplays this help message.\n");
|
||||||
exit(1);
|
exit(1);
|
||||||
|
@ -47,6 +50,7 @@ int main(int argc, char **argv) {
|
||||||
const char *error_msg_part = "load the image";
|
const char *error_msg_part = "load the image";
|
||||||
bool dither = false;
|
bool dither = false;
|
||||||
bool reverse = false;
|
bool reverse = false;
|
||||||
|
bool compress = false;
|
||||||
uint8_t br, bg, bb;
|
uint8_t br, bg, bb;
|
||||||
bool border_set = false;
|
bool border_set = false;
|
||||||
InitializeMagick(*argv);
|
InitializeMagick(*argv);
|
||||||
|
@ -180,6 +184,10 @@ int main(int argc, char **argv) {
|
||||||
reverse = true;
|
reverse = true;
|
||||||
argc--;
|
argc--;
|
||||||
argv++;
|
argv++;
|
||||||
|
} else if (!strcmp(argv[0], "-compress")) {
|
||||||
|
compress = true;
|
||||||
|
argc--;
|
||||||
|
argv++;
|
||||||
} else if (!strcmp(argv[0], "-help")) {
|
} else if (!strcmp(argv[0], "-help")) {
|
||||||
usage();
|
usage();
|
||||||
} else if (!strcmp(argv[0], "-debug")) {
|
} else if (!strcmp(argv[0], "-debug")) {
|
||||||
|
@ -224,15 +232,15 @@ int main(int argc, char **argv) {
|
||||||
if (tcolorcount == 0) tcolorcount = (1 << tbpp);
|
if (tcolorcount == 0) tcolorcount = (1 << tbpp);
|
||||||
try {
|
try {
|
||||||
BitmapX16 bitmap;
|
BitmapX16 bitmap;
|
||||||
|
bitmap.enable_compression(compress);
|
||||||
if (reverse) {
|
if (reverse) {
|
||||||
printf("Loading X16 bitmap file '%s' to be convertedto a PC format...\n", input);
|
printf("Loading X16 bitmap file '%s' to be convertedto a PC format...\n", input);
|
||||||
bitmap.load_x16(input);
|
bitmap.load_x16(input);
|
||||||
printf("Image has %u significant colors starting at %u and is %u bpp\n", bitmap.get_significant(), bitmap.get_significant_start(), bitmap.get_bpp());
|
printf("Image has %u significant colors starting at %u and is %u bpp\n", bitmap.get_significant(), bitmap.get_significant_start(), bitmap.get_bpp());
|
||||||
PaletteEntry border = bitmap.get_palette_entry(bitmap.get_border_color(), true);
|
|
||||||
printf("Border color: %s\n", border.to_string().c_str());
|
|
||||||
} else {
|
} else {
|
||||||
printf("Loading PC image file '%s' to be converted to the X16 bitmap format...\n", input);
|
printf("Loading PC image file '%s' to be converted to the X16 bitmap format...\n", input);
|
||||||
printf("Using at most %u colors (excluding border color) at %u bpp\n", tcolorcount, tbpp);
|
printf("Using at most %u colors (excluding border color) at %u bpp\n", tcolorcount, tbpp);
|
||||||
|
if (bitmap.compression_enabled()) printf("Compression enabled\n");
|
||||||
bitmap.load_pc(input);
|
bitmap.load_pc(input);
|
||||||
}
|
}
|
||||||
if (tw != 0 && th != 0) {
|
if (tw != 0 && th != 0) {
|
||||||
|
|
31
meson.build
31
meson.build
|
@ -1,19 +1,44 @@
|
||||||
project('graphicsconverter', 'cpp',
|
project('graphicsconverter', ['cpp', 'c'],
|
||||||
version : '0.1',
|
version : '0.1',
|
||||||
default_options : ['warning_level=3',
|
default_options : ['warning_level=3',
|
||||||
'cpp_std=c++17',
|
'cpp_std=c++17',
|
||||||
'default_library=static'])
|
'default_library=static'])
|
||||||
|
|
||||||
deps = [
|
deps = [
|
||||||
dependency('Magick++', version : '>=6.9.11')
|
dependency('Magick++', version : '>=6.9.11'),
|
||||||
|
dependency('openmp')
|
||||||
]
|
]
|
||||||
srcs = [
|
srcs = [
|
||||||
'palette.cpp',
|
'palette.cpp',
|
||||||
'bitmapx16.cpp',
|
'bitmapx16.cpp',
|
||||||
'main.cpp'
|
'main.cpp',
|
||||||
|
'lzsa/src/dictionary.c',
|
||||||
|
'lzsa/src/expand_block_v1.c',
|
||||||
|
'lzsa/src/expand_block_v2.c',
|
||||||
|
'lzsa/src/expand_context.c',
|
||||||
|
'lzsa/src/expand_inmem.c',
|
||||||
|
'lzsa/src/expand_streaming.c',
|
||||||
|
'lzsa/src/frame.c',
|
||||||
|
'lzsa/src/matchfinder.c',
|
||||||
|
'lzsa/src/shrink_block_v1.c',
|
||||||
|
'lzsa/src/shrink_block_v2.c',
|
||||||
|
'lzsa/src/shrink_context.c',
|
||||||
|
'lzsa/src/shrink_inmem.c',
|
||||||
|
'lzsa/src/shrink_streaming.c',
|
||||||
|
'lzsa/src/stream.c',
|
||||||
|
'lzsa/src/libdivsufsort/lib/divsufsort_utils.c',
|
||||||
|
'lzsa/src/libdivsufsort/lib/divsufsort.c',
|
||||||
|
'lzsa/src/libdivsufsort/lib/sssort.c',
|
||||||
|
'lzsa/src/libdivsufsort/lib/trsort.c'
|
||||||
|
]
|
||||||
|
incdirs = [
|
||||||
|
'lzsa/src',
|
||||||
|
'lzsa/src/libdivsufsort/include'
|
||||||
]
|
]
|
||||||
exe = executable('b16converter', srcs,
|
exe = executable('b16converter', srcs,
|
||||||
install : true,
|
install : true,
|
||||||
|
include_directories: incdirs,
|
||||||
|
c_args: ['-DHAVE_CONFIG_H=1','-D__STDC_LIMIT_MACROS','-D__STDC_CONSTANT_MACROS','-D__STDC_FORMAT_MACROS'],
|
||||||
dependencies : deps)
|
dependencies : deps)
|
||||||
|
|
||||||
test('basic', exe)
|
test('basic', exe)
|
||||||
|
|
43
test.sh
43
test.sh
|
@ -31,11 +31,12 @@ resize=()
|
||||||
enable_defaults=1
|
enable_defaults=1
|
||||||
dither=1
|
dither=1
|
||||||
reverse=1
|
reverse=1
|
||||||
|
enable_compression=1
|
||||||
enable_reverse=1
|
enable_reverse=1
|
||||||
enable_dither=1
|
enable_dither=1
|
||||||
debug_flags=""
|
debug_flags=""
|
||||||
outdir="testout"
|
outdir="testout"
|
||||||
OPTIONS=$(getopt -o "b:hp:i:r:no:d:" --long "help,use-program:,input-image:,output-bpp:,resize:,no-defaults,output-dir:,debug:,no-reverse,no-dither" -- "$@")
|
OPTIONS=$(getopt -o "b:hp:i:r:no:d:" --long "help,use-program:,input-image:,output-bpp:,resize:,no-defaults,output-dir:,debug:,no-reverse,no-dither,no-compress" -- "$@")
|
||||||
if [ $? != 0 ]; then
|
if [ $? != 0 ]; then
|
||||||
echo "Getopt error."
|
echo "Getopt error."
|
||||||
usage
|
usage
|
||||||
|
@ -79,6 +80,10 @@ while [ -n "$1" ]; do
|
||||||
enable_dither=0
|
enable_dither=0
|
||||||
shift
|
shift
|
||||||
;;
|
;;
|
||||||
|
--no-compress)
|
||||||
|
enable_compression=0
|
||||||
|
shift
|
||||||
|
;;
|
||||||
-d|--debug)
|
-d|--debug)
|
||||||
debug_flags="${debug_flags}$2"
|
debug_flags="${debug_flags}$2"
|
||||||
shift 2
|
shift 2
|
||||||
|
@ -110,20 +115,30 @@ run() {
|
||||||
for img in "${images[@]}"; do
|
for img in "${images[@]}"; do
|
||||||
for bpp in "${bpps[@]}"; do
|
for bpp in "${bpps[@]}"; do
|
||||||
for size in "${resize[@]}"; do
|
for size in "${resize[@]}"; do
|
||||||
width="$(echo -n "$size" | cut -dx -f1)"
|
for compressflag in -compress ""; do
|
||||||
height="$(echo -n "$size" | cut -dx -f2)"
|
width="$(echo -n "$size" | cut -dx -f1)"
|
||||||
name="$(basename "$img" | sed 's/\.png$//')"
|
height="$(echo -n "$size" | cut -dx -f2)"
|
||||||
name="$(printf "%s.%sP.%sB" "$name" "$width" "$bpp")"
|
name="$(basename "$img" | sed 's/\.png$//')"
|
||||||
run "$converter" -in "$img" -out "$outdir/$name.BMX" -bpp "$bpp" -resize "$width" "$height" -border 15 0 15 -debug "$debug_flags"
|
name="$(printf "%s.%sP.%sB" "$name" "$width" "$bpp")"
|
||||||
if [ $enable_dither -ne 0 ]; then
|
extraflags=()
|
||||||
run "$converter" -in "$img" -out "$outdir/$name.D.BMX" -bpp "$bpp" -resize "$width" "$height" -dither -border 15 0 15 -debug "$debug_flags"
|
if [ -n "$compressflag" ]; then
|
||||||
fi
|
if [ $enable_compression -eq 0 ]; then
|
||||||
if [ $enable_reverse -ne 0 ]; then
|
continue
|
||||||
run "$converter" -reverse -in "$outdir/$name.BMX" -out "$outdir/$name.PNG" -resize "$width" "$height" -debug "$debug_flags"
|
fi
|
||||||
if [ $enable_dither -ne 0 ]; then
|
extraflags+=( "$compressflag" )
|
||||||
run "$converter" -reverse -in "$outdir/$name.D.BMX" -out "$outdir/$name.D.PNG" -resize "$width" "$height" -dither -debug "$debug_flags"
|
name+=".C"
|
||||||
fi
|
fi
|
||||||
fi
|
run "$converter" "${extraflags[@]}" -in "$img" -out "$outdir/$name.BMX" -bpp "$bpp" -resize "$width" "$height" -border 15 0 15 -debug "$debug_flags"
|
||||||
|
if [ $enable_dither -ne 0 ]; then
|
||||||
|
run "$converter" "${extraflags[@]}" -in "$img" -out "$outdir/$name.D.BMX" -bpp "$bpp" -resize "$width" "$height" -dither -border 15 0 15 -debug "$debug_flags"
|
||||||
|
fi
|
||||||
|
if [ $enable_reverse -ne 0 ]; then
|
||||||
|
run "$converter" "${extraflags[@]}" -reverse -in "$outdir/$name.BMX" -out "$outdir/$name.PNG" -resize "$width" "$height" -debug "$debug_flags"
|
||||||
|
if [ $enable_dither -ne 0 ]; then
|
||||||
|
run "$converter" "${extraflags[@]}" -reverse -in "$outdir/$name.D.BMX" -out "$outdir/$name.D.PNG" -resize "$width" "$height" -dither -debug "$debug_flags"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
done
|
||||||
done
|
done
|
||||||
done
|
done
|
||||||
done
|
done
|
||||||
|
|
Loading…
Reference in a new issue