diff options
-rw-r--r-- | src/map.c | 195 |
1 files changed, 181 insertions, 14 deletions
@@ -1,5 +1,5 @@ /* - * Copyright (C) 2021 P. J. McDermott + * Copyright (C) 2013, 2021 P. J. McDermott * * This file is part of Dodge Balls * @@ -21,6 +21,8 @@ #include <stdio.h> #include <stdlib.h> #include <string.h> +#include "base64.h" +#include "compression.h" #include "defs.h" #include "dirs.h" #include "map.h" @@ -28,18 +30,28 @@ #include "output.h" #include "xml.h" +struct db_map_layer { + Uint32 *tiles; + char *encoding; + char *compression; + char *raw_data; + struct db_map_layer *next; +}; + struct db_map { - char *game_id; - int w; - int h; - int tw; - int th; - Uint8 bg_r; - Uint8 bg_g; - Uint8 bg_b; - int fr; - struct db_tileset *tileset_head; - struct db_tileset *tileset_tail; + char *game_id; + int w; + int h; + int tw; + int th; + Uint8 bg_r; + Uint8 bg_g; + Uint8 bg_b; + int fr; + struct db_tileset *tileset_head; + struct db_tileset *tileset_tail; + struct db_map_layer *layer_head; + struct db_map_layer *layer_tail; }; static char * @@ -175,6 +187,162 @@ _db_tmx_tileset_end(void *pv, const char *name) } static void XMLCALL +_db_tmx_cdata(void *pv, const char *s, int len) +{ + XML_Parser p; + struct db_map *map; + char *s_z; + char *s_z_trimmed; + char *s_z_trimmed_end; + char *old_data; + + p = (XML_Parser) pv; + map = db_xml_node_peek(p); + + s_z = s_z_trimmed = strndup(s, len); + + while(isspace(*s_z_trimmed)) { + ++s_z_trimmed; + --len; + } + if (*s_z_trimmed == '\0') { + db_dbg(" (null)"); + free(s_z); + return; + } + s_z_trimmed_end = s_z_trimmed + len - 1; + while (s_z_trimmed_end > s_z_trimmed && isspace(*s_z_trimmed_end)) { + --s_z_trimmed_end; + } + *(s_z_trimmed_end + 1) = '\0'; + + db_dbg(" %s", s_z_trimmed); + if (map->layer_tail->raw_data != NULL && + *map->layer_tail->raw_data != '\0') { + old_data = map->layer_tail->raw_data; + sprintf(map->layer_tail->raw_data, "%s %s", + map->layer_tail->raw_data, strdup(s_z_trimmed)); + free(old_data); + } else { + map->layer_tail->raw_data = strdup(s_z_trimmed); + } + + free(s_z); +} + +static void XMLCALL +_db_tmx_data_end(void *pv, const char *name) +{ + XML_Parser p; + struct db_map *map; + size_t decoded_len; + size_t decomp_len; + char *decoded_buf; + char *decomp_buf; + size_t i; + + db_dbg(" </%s> (data)", name); + + p = (XML_Parser) pv; + map = db_xml_node_peek(p); + + if (db_xml_check_tag(name, "data")) { + decoded_len = strlen(map->layer_tail->raw_data); + decomp_len = 4 * map->w* map->h; + db_dbg(" Expected map data size: %zu", decomp_len); + decoded_buf = malloc(decoded_len + 1); + if (decoded_buf == NULL) { + db_warn(" Failed to allocate layer data buffer"); + XML_StopParser(p, XML_FALSE); + return; + } + if (strcmp(map->layer_tail->encoding, "base64") == 0) { + db_dbg(" Decoding base 64 layer data..."); + db_base64_decode(map->layer_tail->raw_data, + decoded_buf, decoded_len + 1); + } + if (map->layer_tail->compression == NULL) { + db_dbg(" Layer data already decompressed"); + decomp_buf = decoded_buf; + } else if (strcmp(map->layer_tail->compression, "zlib") == 0) { + db_dbg(" Decompressing layer data with zlib..."); + decomp_buf = malloc(decomp_len); + if (decomp_buf == NULL) { + db_err("Failed to allocate layer data buffer"); + free(decoded_buf); + XML_StopParser(p, XML_FALSE); + return; + } + db_decompress(decoded_buf, decoded_len, + decomp_buf, decomp_len); + free(decoded_buf); + } else if (strcmp(map->layer_tail->compression, "gzip") == 0) { + db_dbg(" Decompressing layer data with gzip..."); + decomp_buf = malloc(decomp_len); + if (decomp_buf == NULL) { + db_err("Failed to allocate layer data buffer"); + free(decoded_buf); + XML_StopParser(p, XML_FALSE); + return; + } + db_decompress(decoded_buf, decoded_len, + decomp_buf, decomp_len); + free(decoded_buf); + } else { + /* This should never happen. This branch exists only to + * silence GCC's maybe-uninitialized warning on + * decomp_buf below. */ + return; + } + free(map->layer_tail->raw_data); + map->layer_tail->tiles = malloc(decomp_len); + for (i = 0; i < decomp_len / 4; ++i) { + /* Convert each tile GID to the system's byte order. */ + map->layer_tail->tiles[i] = + (decomp_buf[i * 4 + 0] & 0xFF << 000) | + (decomp_buf[i * 4 + 1] & 0xFF << 010) | + (decomp_buf[i * 4 + 2] & 0xFF << 020) | + (decomp_buf[i * 4 + 3] & 0xFF << 030); + } + free(decomp_buf); + db_xml_node_pop(p); + } else { + db_xml_unexpected_end_tag(p, name, "data"); + } +} + +static void XMLCALL +_db_tmx_data_start(void *pv, const char *name, const char **attr) +{ + XML_Parser p; + struct db_map *map; + + db_dbg(" <%s> (data)", name); + + p = (XML_Parser) pv; + map = db_xml_node_peek(p); + + if (db_xml_check_tag(name, "data")) { + map->layer_tail = calloc(1, sizeof(*map->layer_tail)); + if (map->layer_tail == NULL) { + XML_StopParser(p, XML_FALSE); + return; + } + if (map->layer_head == NULL) { + map->layer_head = map->layer_tail; + } + db_xml_get_string_attr(p, attr, "encoding", + &map->layer_tail->encoding, 1); + db_xml_get_string_attr(p, attr, "compression", + &map->layer_tail->compression, 0); + db_xml_node_push(p, map, _db_tmx_invalid_start, + _db_tmx_data_end, _db_tmx_cdata); + } else { + db_xml_unexpected_start_tag(p, name, "data"); + } +} + +static void XMLCALL _db_tmx_layer_end(void *pv, const char *name) { XML_Parser p; @@ -239,8 +407,7 @@ _db_tmx_map_el_start(void *pv, const char *name, const char **attr) db_xml_node_push(p, map, _db_tmx_invalid_start, _db_tmx_tileset_end, NULL); } else if (db_xml_check_tag(name, "layer")) { - /* TODO: data */ - db_xml_node_push(p, map, _db_tmx_invalid_start, + db_xml_node_push(p, map, _db_tmx_data_start, _db_tmx_layer_end, NULL); } else if (db_xml_check_tag(name, "objectgroup")) { /* TODO: object */ |