Files @ r17613:a9b2554a5d79
Branch filter:

Location: cpp/openttd-patchpack/source/src/spriteloader/png.cpp - annotation

rubidium
(svn r22387) -Fix-ish [FS#4601]: Windows' recv seems to return "graceful closed" before having passed the remaining buffer which causes OpenTTD to think all connections are "incorrectly" terminated, i.e. without the "I'm leaving" packet from the client. So let the client wait a tiny bit after sending the "I'm leaving" packet and before gracefully closing the connection
r6896:34553e48985a
r6896:34553e48985a
r12768:980ae0491352
r12768:980ae0491352
r12768:980ae0491352
r12768:980ae0491352
r12768:980ae0491352
r12768:980ae0491352
r12768:980ae0491352
r9111:983de9c5a848
r6896:34553e48985a
r7098:956b83134608
r7098:956b83134608
r6896:34553e48985a
r10039:727fb45b0424
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r14195:ea01033bb77f
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r14328:70a41f8a4696
r14328:70a41f8a4696
r14328:70a41f8a4696
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r9151:8f9b91035e52
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r11085:c87a330fb4c2
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r14195:ea01033bb77f
r14195:ea01033bb77f
r11202:ec3713bd39fb
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r11085:c87a330fb4c2
r6896:34553e48985a
r11085:c87a330fb4c2
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r11085:c87a330fb4c2
r6907:40e387a550dd
r11085:c87a330fb4c2
r6907:40e387a550dd
r11085:c87a330fb4c2
r6907:40e387a550dd
r11085:c87a330fb4c2
r6907:40e387a550dd
r6896:34553e48985a
r11085:c87a330fb4c2
r15003:fcdf4495a462
r15003:fcdf4495a462
r15003:fcdf4495a462
r15003:fcdf4495a462
r15003:fcdf4495a462
r15003:fcdf4495a462
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r14195:ea01033bb77f
r6896:34553e48985a
r14195:ea01033bb77f
r6896:34553e48985a
r6896:34553e48985a
r14195:ea01033bb77f
r6896:34553e48985a
r14195:ea01033bb77f
r6896:34553e48985a
r6896:34553e48985a
r8757:ce3c6e136100
r6896:34553e48985a
r8757:ce3c6e136100
r11085:c87a330fb4c2
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r8757:ce3c6e136100
r6896:34553e48985a
r8757:ce3c6e136100
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r6896:34553e48985a
r10056:7a18efcd8edb
r6896:34553e48985a
r8374:ffc62b01a0a7
r9390:aab57f0a7820
r9390:aab57f0a7820
r6896:34553e48985a
r6896:34553e48985a
r7098:956b83134608
r7098:956b83134608
/* $Id$ */

/*
 * This file is part of OpenTTD.
 * OpenTTD is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, version 2.
 * OpenTTD is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
 * See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with OpenTTD. If not, see <http://www.gnu.org/licenses/>.
 */

/** @file png.cpp Reading sprites from png files. */

#ifdef WITH_PNG

#include "../stdafx.h"
#include "../fileio_func.h"
#include "../debug.h"
#include "png.hpp"
#include <png.h>

#define PNG_SLOT 62

static void PNGAPI png_my_read(png_structp png_ptr, png_bytep data, png_size_t length)
{
	FioReadBlock(data, length);
}

static void PNGAPI png_my_error(png_structp png_ptr, png_const_charp message)
{
	DEBUG(sprite, 0, "ERROR (libpng): %s - %s", message, (char *)png_get_error_ptr(png_ptr));
	longjmp(png_jmpbuf(png_ptr), 1);
}

static void PNGAPI png_my_warning(png_structp png_ptr, png_const_charp message)
{
	DEBUG(sprite, 0, "WARNING (libpng): %s - %s", message, (char *)png_get_error_ptr(png_ptr));
}

static bool OpenPNGFile(const char *filename, uint32 id, bool mask)
{
	char png_file[MAX_PATH];

	/* Add path separator after 'sprites' if not present */
	const char *sep = (filename[0] == PATHSEPCHAR) ? "" : PATHSEP;
	snprintf(png_file, sizeof(png_file), "sprites%s%s" PATHSEP "%d%s.png", sep, filename, id, mask ? "m" : "");
	if (FioCheckFileExists(png_file)) {
		FioOpenFile(PNG_SLOT, png_file);
		return true;
	}

	/* TODO -- Add TAR support */
	return false;
}

static bool LoadPNG(SpriteLoader::Sprite *sprite, const char *filename, uint32 id, volatile bool mask)
{
	png_byte header[8];
	png_structp png_ptr;
	png_infop info_ptr, end_info;
	uint bit_depth, colour_type;
	uint i, pixelsize;
	SpriteLoader::CommonPixel *dst;

	if (!OpenPNGFile(filename, id, mask)) return mask; // If mask is true, and file not found, continue true anyway, as it isn't a show-stopper

	/* Check the header */
	FioReadBlock(header, 8);
	if (png_sig_cmp(header, 0, 8) != 0) return false;

	/* Create the reader */
	png_ptr = png_create_read_struct(PNG_LIBPNG_VER_STRING, (png_voidp)NULL, png_my_error, png_my_warning);
	if (png_ptr == NULL) return false;

	/* Create initial stuff */
	info_ptr = png_create_info_struct(png_ptr);
	if (info_ptr == NULL) {
		png_destroy_read_struct(&png_ptr, (png_infopp)NULL, (png_infopp)NULL);
		return false;
	}
	end_info = png_create_info_struct(png_ptr);
	if (end_info == NULL) {
		png_destroy_read_struct(&png_ptr, &info_ptr, (png_infopp)NULL);
		return false;
	}

	/* Make sure that upon error, we can clean up graceful */
	if (setjmp(png_jmpbuf(png_ptr))) {
		png_destroy_read_struct(&png_ptr, &info_ptr, &end_info);
		return false;
	}

	/* Read the file */
	png_set_read_fn(png_ptr, NULL, png_my_read);
	png_set_sig_bytes(png_ptr, 8);

	png_read_info(png_ptr, info_ptr);

	if (!mask) {
		/* Read the text chunks */
		png_textp text_ptr;
		int num_text = 0;
		png_get_text(png_ptr, info_ptr, &text_ptr, &num_text);
		if (num_text == 0) DEBUG(misc, 0, "Warning: PNG Sprite '%s/%d.png' doesn't have x_offs and y_offs; expect graphical problems", filename, id);
		for (int i = 0; i < num_text; i++) {
			/* x_offs and y_offs are in the text-chunk of PNG */
			if (strcmp("x_offs", text_ptr[i].key) == 0) sprite->x_offs = strtol(text_ptr[i].text, NULL, 0);
			if (strcmp("y_offs", text_ptr[i].key) == 0) sprite->y_offs = strtol(text_ptr[i].text, NULL, 0);
		}

		sprite->height = png_get_image_height(png_ptr, info_ptr);
		sprite->width  = png_get_image_width(png_ptr, info_ptr);
		sprite->AllocateData(sprite->width * sprite->height);
	}

	bit_depth  = png_get_bit_depth(png_ptr, info_ptr);
	colour_type = png_get_color_type(png_ptr, info_ptr);

	if (mask && (bit_depth != 8 || colour_type != PNG_COLOR_TYPE_PALETTE)) {
		DEBUG(misc, 0, "Ignoring mask for SpriteID %d as it isn't a 8 bit palette image", id);
		return true;
	}

	if (!mask) {
		if (bit_depth == 16) png_set_strip_16(png_ptr);

		if (colour_type == PNG_COLOR_TYPE_PALETTE) {
			png_set_palette_to_rgb(png_ptr);
			colour_type = PNG_COLOR_TYPE_RGB;
		}
		if (colour_type == PNG_COLOR_TYPE_GRAY || colour_type == PNG_COLOR_TYPE_GRAY_ALPHA) {
			png_set_gray_to_rgb(png_ptr);
			colour_type = PNG_COLOR_TYPE_RGB;
		}

		if (colour_type == PNG_COLOR_TYPE_RGB) {
			if (png_get_valid(png_ptr, info_ptr, PNG_INFO_tRNS)) {
				/* Create an alpha channel when there is a tRNS chunk */
				png_set_tRNS_to_alpha(png_ptr);
			} else {
				png_set_filler(png_ptr, 0xff, PNG_FILLER_AFTER);
			}
		}

		pixelsize = sizeof(uint32);
	} else {
		pixelsize = sizeof(uint8);
	}

	png_bytep row_pointer = AllocaM(png_byte, png_get_image_width(png_ptr, info_ptr) * pixelsize);

	for (i = 0; i < png_get_image_height(png_ptr, info_ptr); i++) {
		png_read_row(png_ptr, row_pointer, NULL);

		dst = sprite->data + i * png_get_image_width(png_ptr, info_ptr);

		for (uint x = 0; x < png_get_image_width(png_ptr, info_ptr); x++) {
			if (mask) {
				if (row_pointer[x * sizeof(uint8)] != 0) {
					dst[x].r = 0;
					dst[x].g = 0;
					dst[x].b = 0;
					/* Alpha channel is used from the original image (to allow transparency in remap colours) */
					dst[x].m = row_pointer[x * sizeof(uint8)];
				}
			} else {
				dst[x].r = row_pointer[x * sizeof(uint32) + 0];
				dst[x].g = row_pointer[x * sizeof(uint32) + 1];
				dst[x].b = row_pointer[x * sizeof(uint32) + 2];
				dst[x].a = row_pointer[x * sizeof(uint32) + 3];
				dst[x].m = 0;
			}
		}
	}

	png_destroy_read_struct(&png_ptr, &info_ptr, &end_info);

	return true;
}

bool SpriteLoaderPNG::LoadSprite(SpriteLoader::Sprite *sprite, uint8 file_slot, size_t file_pos, SpriteType sprite_type)
{
	const char *filename = FioGetFilename(file_slot);
	if (!LoadPNG(sprite, filename, (uint32)file_pos, false)) return false;
	if (!LoadPNG(sprite, filename, (uint32)file_pos, true)) return false;
	return true;
}

#endif /* WITH_PNG */