/* Copyright (c) 2013-2014 Jeffrey Pfau * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include #ifdef USE_PNG #include static void _pngWrite(png_structp png, png_bytep buffer, png_size_t size) { struct VFile* vf = png_get_io_ptr(png); size_t written = vf->write(vf, buffer, size); if (written != size) { png_error(png, "Could not write PNG"); } } static void _pngRead(png_structp png, png_bytep buffer, png_size_t size) { struct VFile* vf = png_get_io_ptr(png); size_t read = vf->read(vf, buffer, size); if (read != size) { png_error(png, "Could not read PNG"); } } png_structp PNGWriteOpen(struct VFile* source) { png_structp png = png_create_write_struct(PNG_LIBPNG_VER_STRING, 0, 0, 0); if (!png) { return 0; } if (setjmp(png_jmpbuf(png))) { png_destroy_write_struct(&png, 0); return 0; } png_set_write_fn(png, source, _pngWrite, 0); return png; } png_infop PNGWriteHeader(png_structp png, unsigned width, unsigned height) { png_infop info = png_create_info_struct(png); if (!info) { return 0; } if (setjmp(png_jmpbuf(png))) { return 0; } png_set_IHDR(png, info, width, height, 8, PNG_COLOR_TYPE_RGB, PNG_INTERLACE_NONE, PNG_COMPRESSION_TYPE_BASE, PNG_FILTER_TYPE_BASE); png_write_info(png, info); return info; } png_infop PNGWriteHeader8(png_structp png, unsigned width, unsigned height) { png_infop info = png_create_info_struct(png); if (!info) { return 0; } if (setjmp(png_jmpbuf(png))) { return 0; } png_set_IHDR(png, info, width, height, 8, PNG_COLOR_TYPE_PALETTE, PNG_INTERLACE_NONE, PNG_COMPRESSION_TYPE_BASE, PNG_FILTER_TYPE_BASE); return info; } bool PNGWritePalette(png_structp png, png_infop info, const uint32_t* palette, unsigned entries) { if (!palette || !entries) { return false; } if (setjmp(png_jmpbuf(png))) { return false; } png_color colors[256]; png_byte trans[256]; unsigned i; for (i = 0; i < entries && i < 256; ++i) { colors[i].red = palette[i]; colors[i].green = palette[i] >> 8; colors[i].blue = palette[i] >> 16; trans[i] = palette[i] >> 24; } png_set_PLTE(png, info, colors, entries); png_set_tRNS(png, info, trans, entries, NULL); png_write_info(png, info); return true; } bool PNGWritePixels(png_structp png, unsigned width, unsigned height, unsigned stride, const void* pixels) { png_bytep row = malloc(sizeof(png_byte) * width * 3); if (!row) { return false; } const png_byte* pixelData = pixels; if (setjmp(png_jmpbuf(png))) { free(row); return false; } unsigned i; for (i = 0; i < height; ++i) { unsigned x; for (x = 0; x < width; ++x) { #ifdef COLOR_16_BIT uint16_t c = ((uint16_t*) pixelData)[stride * i + x]; #ifdef COLOR_5_6_5 row[x * 3] = (c >> 8) & 0xF8; row[x * 3 + 1] = (c >> 3) & 0xFC; row[x * 3 + 2] = (c << 3) & 0xF8; #else row[x * 3] = (c >> 7) & 0xF8; row[x * 3 + 1] = (c >> 2) & 0xF8; row[x * 3 + 2] = (c << 3) & 0xF8; #endif #else #ifdef __BIG_ENDIAN__ row[x * 3] = pixelData[stride * i * 4 + x * 4 + 3]; row[x * 3 + 1] = pixelData[stride * i * 4 + x * 4 + 2]; row[x * 3 + 2] = pixelData[stride * i * 4 + x * 4 + 1]; #else row[x * 3] = pixelData[stride * i * 4 + x * 4]; row[x * 3 + 1] = pixelData[stride * i * 4 + x * 4 + 1]; row[x * 3 + 2] = pixelData[stride * i * 4 + x * 4 + 2]; #endif #endif } png_write_row(png, row); } free(row); return true; } bool PNGWritePixels8(png_structp png, unsigned width, unsigned height, unsigned stride, const void* pixels) { UNUSED(width); const png_byte* pixelData = pixels; if (setjmp(png_jmpbuf(png))) { return false; } unsigned i; for (i = 0; i < height; ++i) { png_write_row(png, &pixelData[stride * i]); } return true; } bool PNGWriteCustomChunk(png_structp png, const char* name, size_t size, void* data) { char realName[5]; strncpy(realName, name, 4); realName[0] = tolower((int) realName[0]); realName[1] = tolower((int) realName[1]); realName[4] = '\0'; if (setjmp(png_jmpbuf(png))) { return false; } png_write_chunk(png, (png_bytep) realName, data, size); return true; } void PNGWriteClose(png_structp png, png_infop info) { if (!setjmp(png_jmpbuf(png))) { png_write_end(png, info); } png_destroy_write_struct(&png, &info); } bool isPNG(struct VFile* source) { png_byte header[PNG_HEADER_BYTES]; if (source->read(source, header, PNG_HEADER_BYTES) < PNG_HEADER_BYTES) { return false; } return !png_sig_cmp(header, 0, PNG_HEADER_BYTES); } png_structp PNGReadOpen(struct VFile* source, unsigned offset) { png_structp png = png_create_read_struct(PNG_LIBPNG_VER_STRING, 0, 0, 0); if (!png) { return 0; } if (setjmp(png_jmpbuf(png))) { png_destroy_read_struct(&png, 0, 0); return 0; } png_set_read_fn(png, source, _pngRead); png_set_sig_bytes(png, offset); return png; } bool PNGInstallChunkHandler(png_structp png, void* context, ChunkHandler handler, const char* chunkName) { if (setjmp(png_jmpbuf(png))) { return false; } png_set_read_user_chunk_fn(png, context, handler); int len = strlen(chunkName); int chunks = 0; char* chunkList = strdup(chunkName); int i; for (i = 4; i <= len; i += 5) { chunkList[i] = '\0'; ++chunks; } png_set_keep_unknown_chunks(png, PNG_HANDLE_CHUNK_ALWAYS, (png_bytep) chunkList, chunks); free(chunkList); return true; } bool PNGReadHeader(png_structp png, png_infop info) { if (setjmp(png_jmpbuf(png))) { return false; } png_read_info(png, info); return true; } bool PNGIgnorePixels(png_structp png, png_infop info) { if (setjmp(png_jmpbuf(png))) { return false; } unsigned height = png_get_image_height(png, info); unsigned i; for (i = 0; i < height; ++i) { png_read_row(png, 0, 0); } return true; } bool PNGReadPixels(png_structp png, png_infop info, void* pixels, unsigned width, unsigned height, unsigned stride) { if (setjmp(png_jmpbuf(png))) { return false; } uint8_t* pixelData = pixels; unsigned pngHeight = png_get_image_height(png, info); if (height < pngHeight) { pngHeight = height; } unsigned pngWidth = png_get_image_width(png, info); if (width < pngWidth) { pngWidth = width; } unsigned i; png_bytep row = malloc(png_get_rowbytes(png, info)); for (i = 0; i < pngHeight; ++i) { png_read_row(png, row, 0); unsigned x; for (x = 0; x < pngWidth; ++x) { #ifdef COLOR_16_BIT uint16_t c = row[x * 3 + 2] >> 3; #ifdef COLOR_5_6_5 c |= (row[x * 3 + 1] << 3) & 0x7E0; c |= (row[x * 3] << 8) & 0xF800; #else c |= (row[x * 3 + 1] << 2) & 0x3E0; c |= (row[x * 3] << 7) & 0x7C00; #endif ((uint16_t*) pixelData)[stride * i + x] = c; #else #if __BIG_ENDIAN__ pixelData[stride * i * 4 + x * 4 + 3] = row[x * 3]; pixelData[stride * i * 4 + x * 4 + 2] = row[x * 3 + 1]; pixelData[stride * i * 4 + x * 4 + 1] = row[x * 3 + 2]; pixelData[stride * i * 4 + x * 4] = 0xFF; #else pixelData[stride * i * 4 + x * 4] = row[x * 3]; pixelData[stride * i * 4 + x * 4 + 1] = row[x * 3 + 1]; pixelData[stride * i * 4 + x * 4 + 2] = row[x * 3 + 2]; pixelData[stride * i * 4 + x * 4 + 3] = 0xFF; #endif #endif } } free(row); return true; } bool PNGReadFooter(png_structp png, png_infop end) { if (setjmp(png_jmpbuf(png))) { return false; } png_read_end(png, end); return true; } void PNGReadClose(png_structp png, png_infop info, png_infop end) { png_destroy_read_struct(&png, &info, &end); } #endif