/***************************************************************************** * * This file is part of Mapnik (c++ mapping toolkit) * * Copyright (C) 2014 Artem Pavlenko * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA * ***************************************************************************** * * Initially developed by Sandro Santilli for CartoDB * *****************************************************************************/ #include "pgraster_wkb_reader.hpp" // mapnik #include // for datasource_exception #include #include #include #include #include #include #include #include // for box2d // boost #include #include namespace { uint8_t read_uint8(const uint8_t** from) { return *(*from)++; } uint16_t read_uint16(const uint8_t** from, uint8_t littleEndian) { uint16_t ret = 0; if (littleEndian) { ret = (*from)[0] | (*from)[1] << 8; } else { /* big endian */ ret = (*from)[0] << 8 | (*from)[1]; } *from += 2; return ret; } /* int16_t read_int16(const uint8_t** from, uint8_t littleEndian) { assert(NULL != from); return read_uint16(from, littleEndian); } */ double read_float64(const uint8_t** from, uint8_t littleEndian) { union { double d; uint64_t i; } ret; if (littleEndian) { ret.i = (uint64_t) ((*from)[0] & 0xff) | (uint64_t) ((*from)[1] & 0xff) << 8 | (uint64_t) ((*from)[2] & 0xff) << 16 | (uint64_t) ((*from)[3] & 0xff) << 24 | (uint64_t) ((*from)[4] & 0xff) << 32 | (uint64_t) ((*from)[5] & 0xff) << 40 | (uint64_t) ((*from)[6] & 0xff) << 48 | (uint64_t) ((*from)[7] & 0xff) << 56; } else { /* big endian */ ret.i = (uint64_t) ((*from)[7] & 0xff) | (uint64_t) ((*from)[6] & 0xff) << 8 | (uint64_t) ((*from)[5] & 0xff) << 16 | (uint64_t) ((*from)[4] & 0xff) << 24 | (uint64_t) ((*from)[3] & 0xff) << 32 | (uint64_t) ((*from)[2] & 0xff) << 40 | (uint64_t) ((*from)[1] & 0xff) << 48 | (uint64_t) ((*from)[0] & 0xff) << 56; } *from += 8; return ret.d; } uint32_t read_uint32(const uint8_t** from, uint8_t littleEndian) { uint32_t ret = 0; if (littleEndian) { ret = (uint32_t) ((*from)[0] & 0xff) | (uint32_t) ((*from)[1] & 0xff) << 8 | (uint32_t) ((*from)[2] & 0xff) << 16 | (uint32_t) ((*from)[3] & 0xff) << 24; } else { /* big endian */ ret = (uint32_t) ((*from)[3] & 0xff) | (uint32_t) ((*from)[2] & 0xff) << 8 | (uint32_t) ((*from)[1] & 0xff) << 16 | (uint32_t) ((*from)[0] & 0xff) << 24; } *from += 4; return ret; } int32_t read_int32(const uint8_t** from, uint8_t littleEndian) { return read_uint32(from, littleEndian); } float read_float32(const uint8_t** from, uint8_t littleEndian) { union { float f; uint32_t i; } ret; ret.i = read_uint32(from, littleEndian); return ret.f; } typedef enum { PT_1BB=0, /* 1-bit boolean */ PT_2BUI=1, /* 2-bit unsigned integer */ PT_4BUI=2, /* 4-bit unsigned integer */ PT_8BSI=3, /* 8-bit signed integer */ PT_8BUI=4, /* 8-bit unsigned integer */ PT_16BSI=5, /* 16-bit signed integer */ PT_16BUI=6, /* 16-bit unsigned integer */ PT_32BSI=7, /* 32-bit signed integer */ PT_32BUI=8, /* 32-bit unsigned integer */ PT_32BF=10, /* 32-bit float */ PT_64BF=11, /* 64-bit float */ PT_END=13 } rt_pixtype; #define BANDTYPE_FLAGS_MASK 0xF0 #define BANDTYPE_PIXTYPE_MASK 0x0F #define BANDTYPE_FLAG_OFFDB (1<<7) #define BANDTYPE_FLAG_HASNODATA (1<<6) #define BANDTYPE_FLAG_ISNODATA (1<<5) #define BANDTYPE_FLAG_RESERVED3 (1<<4) #define BANDTYPE_PIXTYPE_MASK 0x0F #define BANDTYPE_PIXTYPE(x) ((x)&BANDTYPE_PIXTYPE_MASK) #define BANDTYPE_IS_OFFDB(x) ((x)&BANDTYPE_FLAG_OFFDB) #define BANDTYPE_HAS_NODATA(x) ((x)&BANDTYPE_FLAG_HASNODATA) #define BANDTYPE_IS_NODATA(x) ((x)&BANDTYPE_FLAG_ISNODATA) } using mapnik::box2d; template void read_data_band(mapnik::raster_ptr raster, uint16_t width, uint16_t height, bool hasnodata, T reader) { mapnik::image_data_32 & image = raster->data_; // Start with plain white (ABGR or RGBA depending on endiannes) // TODO: set to transparent instead? image.set(0xffffffff); raster->premultiplied_alpha_ = true; float* data = (float*)image.getBytes(); double val; val = reader(); // nodata value, need to read anyway if ( hasnodata ) raster->set_nodata(val); for (int y=0; ydata_; // Start with all zeroes image.set(0); uint8_t type = read_uint8(&ptr_); int pixtype = BANDTYPE_PIXTYPE(type); int offline = BANDTYPE_IS_OFFDB(type) ? 1 : 0; int hasnodata = BANDTYPE_HAS_NODATA(type) ? 1 : 0; MAPNIK_LOG_DEBUG(pgraster) << "pgraster_wkb_reader: band type:" << pixtype << " offline:" << offline << " hasnodata:" << hasnodata; if ( offline ) { MAPNIK_LOG_WARN(pgraster) << "pgraster_wkb_reader: offline band " " unsupported"; return; } MAPNIK_LOG_DEBUG(pgraster) << "pgraster_wkb_reader: reading " << height_ << "x" << width_ << " pixels"; switch (pixtype) { case PT_1BB: case PT_2BUI: case PT_4BUI: // all <8BPP values are wrote in full bytes anyway case PT_8BSI: // mapnik does not support signed anyway case PT_8BUI: read_data_band(raster, width_, height_, hasnodata, boost::bind(read_uint8, &ptr_)); break; case PT_16BSI: // mapnik does not support signed anyway case PT_16BUI: read_data_band(raster, width_, height_, hasnodata, boost::bind(read_uint16, &ptr_, endian_)); break; case PT_32BSI: // mapnik does not support signed anyway case PT_32BUI: read_data_band(raster, width_, height_, hasnodata, boost::bind(read_uint32, &ptr_, endian_)); break; case PT_32BF: read_data_band(raster, width_, height_, hasnodata, boost::bind(read_float32, &ptr_, endian_)); break; case PT_64BF: read_data_band(raster, width_, height_, hasnodata, boost::bind(read_float64, &ptr_, endian_)); break; default: std::ostringstream err; err << "pgraster_wkb_reader: data band type " << pixtype << " unsupported"; // TODO: accept policy to decide on throw-or-skip ? //MAPNIK_LOG_WARN(pgraster) << err.str(); throw mapnik::datasource_exception(err.str()); } } template void read_grayscale_band(mapnik::raster_ptr raster, uint16_t width, uint16_t height, bool hasnodata, T reader) { mapnik::image_data_32 & image = raster->data_; // Start with plain white (ABGR or RGBA depending on endiannes) // TODO: set to transparent instead? image.set(0xffffffff); raster->premultiplied_alpha_ = true; int val; uint8_t * data = image.getBytes(); int ps = 4; // sizeof(image_data::pixel_type) int off; val = reader(); // nodata value, need to read anyway if ( hasnodata ) raster->set_nodata(val); for (int y=0; ydata_; // Start with plain white (ABGR or RGBA depending on endiannes) image.set(0xffffffff); //raster->set_nodata(0xffffffff); raster->premultiplied_alpha_ = true; uint8_t nodataval; for (int bn=0; bn PT_8BUI || pixtype < PT_8BSI ) { MAPNIK_LOG_WARN(pgraster) << "pgraster_wkb_reader: band " << bn << " type " << type << " unsupported"; continue; } uint8_t tmp = read_uint8(&ptr_); if ( ! bn ) nodataval = tmp; else if ( tmp != nodataval ) { MAPNIK_LOG_WARN(pgraster) << "pgraster_wkb_reader: band " << bn << " nodataval " << tmp << " != band 0 nodataval " << nodataval; } int ps = 4; // sizeof(image_data::pixel_type) uint8_t * image_data = image.getBytes(); for (int y=0; y ext(ipX,ipY,ipX+(width_*scaleX),ipY+(height_*scaleY)); MAPNIK_LOG_DEBUG(pgraster) << "pgraster_wkb_reader: Raster extent=" << ext; mapnik::raster_ptr raster = std::make_shared(ext, width_, height_, 1.0); if ( bandno_ ) { if ( bandno_ != 1 ) { MAPNIK_LOG_WARN(pgraster) << "pgraster_wkb_reader: " "reading bands other than 1st as indexed is unsupported"; return mapnik::raster_ptr(); } MAPNIK_LOG_DEBUG(pgraster) << "pgraster_wkb_reader: requested band " << bandno_; read_indexed(raster); } else { switch (numBands_) { case 1: read_grayscale(raster); break; case 3: case 4: read_rgba(raster); break; default: std::ostringstream err; err << "pgraster_wkb_reader: raster with " << numBands_ << " bands is not supported, specify a band number"; //MAPNIK_LOG_WARN(pgraster) << err.str(); throw mapnik::datasource_exception(err.str()); return mapnik::raster_ptr(); } } return raster; }