/***************************************************************************** * * This file is part of Mapnik (c++ mapping toolkit) * * Copyright (C) 2015 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 * *****************************************************************************/ // mapnik #include #include #include #pragma GCC diagnostic push #include extern "C" { #include #include } #pragma GCC diagnostic pop // stl #include #include namespace mapnik { struct external_buffer_policy { external_buffer_policy( uint8_t const* data, std::size_t size) : data_(data), size_(size) {} uint8_t const* data() const { return data_; } std::size_t size() const { return size_; } uint8_t const* data_; std::size_t size_; }; struct internal_buffer_policy { internal_buffer_policy(std::size_t size) : data_((size!=0) ? static_cast(::operator new(sizeof(uint8_t) * size)) : 0), size_(size) {} uint8_t * data() const { return data_; } std::size_t size() const { return size_; } ~internal_buffer_policy() { ::operator delete(data_), data_=0; } uint8_t * data_; std::size_t size_; }; template class webp_reader : public image_reader { using buffer_policy_type = T; private: struct config_guard { config_guard(WebPDecoderConfig & config) : config_(config) {} ~config_guard() { WebPFreeDecBuffer(&config_.output); } WebPDecoderConfig & config_; }; std::unique_ptr buffer_; size_t size_; unsigned width_; unsigned height_; bool has_alpha_; public: explicit webp_reader(char const* data, std::size_t size); explicit webp_reader(std::string const& filename); ~webp_reader(); unsigned width() const final; unsigned height() const final; boost::optional > bounding_box() const final; inline bool has_alpha() const final { return has_alpha_; } void read(unsigned x,unsigned y,image_rgba8& image) final; image_any read(unsigned x, unsigned y, unsigned width, unsigned height) final; private: void init(); }; namespace { image_reader* create_webp_reader(char const * data, std::size_t size) { return new webp_reader(data, size); } image_reader* create_webp_reader2(std::string const& filename) { return new webp_reader(filename); } const bool registered = register_image_reader("webp", create_webp_reader); const bool registered2 = register_image_reader("webp", create_webp_reader2); } // ctor template webp_reader::webp_reader(char const* data, std::size_t size) : buffer_(new buffer_policy_type(reinterpret_cast(data), size)), width_(0), height_(0), has_alpha_(false) { init(); } template webp_reader::webp_reader(std::string const& filename) : buffer_(nullptr), size_(0), width_(0), height_(0), has_alpha_(false) { std::ifstream file(filename.c_str(), std::ios::binary); if (!file) { throw image_reader_exception("WEBP: Can't read file:" + filename); } std::streampos beg = file.tellg(); file.seekg (0, std::ios::end); std::streampos end = file.tellg(); std::size_t file_size = end - beg; file.seekg (0, std::ios::beg); auto buffer = std::make_unique(file_size); file.read(reinterpret_cast(buffer->data()), buffer->size()); if (!file) { throw image_reader_exception("WEBP: Failed to read:" + filename); } buffer_ = std::move(buffer); init(); } // dtor template webp_reader::~webp_reader() { // } template void webp_reader::init() { WebPDecoderConfig config; config_guard guard(config); if (!WebPInitDecoderConfig(&config)) { throw image_reader_exception("WEBP reader: WebPInitDecoderConfig failed"); } if (WebPGetFeatures(buffer_->data(), buffer_->size(), &config.input) == VP8_STATUS_OK) { width_ = config.input.width; height_ = config.input.height; has_alpha_ = config.input.has_alpha; } else { throw image_reader_exception("WEBP reader: WebPGetFeatures failed"); } } template unsigned webp_reader::width() const { return width_; } template unsigned webp_reader::height() const { return height_; } template boost::optional > webp_reader::bounding_box() const { return boost::optional >(); } template void webp_reader::read(unsigned x0, unsigned y0,image_rgba8& image) { WebPDecoderConfig config; config_guard guard(config); if (!WebPInitDecoderConfig(&config)) { throw image_reader_exception("WEBP reader: WebPInitDecoderConfig failed"); } config.options.use_cropping = 1; config.options.crop_left = x0; config.options.crop_top = y0; config.options.crop_width = std::min(static_cast(width_ - x0), image.width()); config.options.crop_height = std::min(static_cast(height_ - y0), image.height()); if (WebPGetFeatures(buffer_->data(), buffer_->size(), &config.input) != VP8_STATUS_OK) { throw image_reader_exception("WEBP reader: WebPGetFeatures failed"); } config.output.colorspace = MODE_RGBA; config.output.u.RGBA.rgba = reinterpret_cast(image.bytes()); config.output.u.RGBA.stride = 4 * image.width(); config.output.u.RGBA.size = image.width() * image.height() * 4; config.output.is_external_memory = 1; if (WebPDecode(buffer_->data(), buffer_->size(), &config) != VP8_STATUS_OK) { throw image_reader_exception("WEBP reader: WebPDecode failed"); } } template image_any webp_reader::read(unsigned x, unsigned y, unsigned width, unsigned height) { image_rgba8 data(width,height); read(x, y, data); return image_any(std::move(data)); } }