/* This file is part of ksquirrel-libs (http://ksquirrel.sf.net) Copyright (c) 2005 Dmitry Baryshev This library is free software; you can redistribute it and/or modify it under the terms of the GNU Library General Public License as published by the Free Software Foundation; either version 2 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 Library General Public License for more details. You should have received a copy of the GNU Library General Public License along with this library; see the file COPYING. If not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include #include #include #include #if defined(HAVE_STDINT_H) #if !defined(__STDC_LIMIT_MACROS) #define __STDC_LIMIT_MACROS #endif #include #endif #include "ksquirrel-libs/fmt_types.h" #include "ksquirrel-libs/fileio.h" #undef PACKAGE #undef PACKAGE_BUGREPORT #undef PACKAGE_NAME #undef PACKAGE_STRING #undef PACKAGE_TARNAME #undef PACKAGE_VERSION #undef VERSION #include #include "fmt_codec_jpeg2000_defs.h" #include "fmt_codec_jpeg2000.h" #include "ksquirrel-libs/fmt_utils.h" #include "ksquirrel-libs/error.h" #include "../xpm/codec_jpeg2000.xpm" /* * * JPEG 2000 standard supports lossy and lossless compression of * single-component (e.g., grayscale) and multicomponent (e.g., color) * imagery. * */ static bool jasperInitialized = false; static void initializeJasper() { #if (JAS_VERSION_MAJOR >= 3) jas_conf_clear(); // Limit JasPer memory usage to at most 512 MB size_t memoryLimit = (512 * 1024) * 1024; size_t jasperTotalMem = jas_get_total_mem_size(); if (!jasperTotalMem) { jasperTotalMem = JAS_DEFAULT_MAX_MEM_USAGE; } memoryLimit = (memoryLimit < jasperTotalMem) ? memoryLimit : jasperTotalMem; jas_conf_set_max_mem_usage(memoryLimit); if (!jas_init_library()) { if (!jas_init_thread()) { jasperInitialized = true; } else { jas_cleanup_library(); } } #else if (!jas_init()) { jasperInitialized = true; } #endif } static void cleanupJasper() { #if (JAS_VERSION_MAJOR >= 3) if (jasperInitialized) { jas_cleanup_thread(); jas_cleanup_library(); } #else if (jasperInitialized) { jas_cleanup(); } #endif } fmt_codec::fmt_codec() : fmt_codec_base() { initializeJasper(); } fmt_codec::~fmt_codec() { cleanupJasper(); } void fmt_codec::options(codec_options *o) { o->version = "0.4.0"; o->name = "JPEG 2000"; o->filter = "*.jp2 *.j2k "; // mime is "....\152\120\040\040", // but some jp2 files don't have this mime header (why ?) // => o->mime is empty o->mime = ""; o->mimetype = "image/jp2"; o->config = ""; o->pixmap = codec_jpeg2000; o->readable = true; o->canbemultiple = false; o->writestatic = false; o->writeanimated = false; o->needtempfile = false; } s32 fmt_codec::read_init(const std::string &file) { gs.image = 0; gs.altimage = 0; gs.data[0] = 0; gs.data[1] = 0; gs.data[2] = 0; if (!jasperInitialized) { return SQE_NOTOK; } in = jas_stream_fopen(file.c_str(), "rb"); if(!in) return SQE_R_NOFILE; currentImage = -1; read_error = false; finfo.animated = false; return SQE_OK; } s32 fmt_codec::read_next() { currentImage++; if(currentImage) return SQE_NOTOK; fmt_image image; gs.image = jas_image_decode(in, -1, 0); jas_stream_close(in); if(!gs.image) return SQE_R_NOMEMORY; s32 family = jas_clrspc_fam(jas_image_clrspc(gs.image)); if(!convert_colorspace()) return SQE_R_BADFILE; jas_image_destroy(gs.image); gs.image = gs.altimage; gs.altimage = 0; image.w = jas_image_width(gs.image); image.h = jas_image_height(gs.image); switch(family) { case JAS_CLRSPC_FAM_RGB: image.colorspace = "RGB"; image.bpp = 24; break; case JAS_CLRSPC_FAM_YCBCR: image.colorspace = "YCbCr"; image.bpp = 24; break; case JAS_CLRSPC_FAM_GRAY: image.colorspace = "Grayscale"; image.bpp = 8; break; case JAS_CLRSPC_FAM_LAB: image.colorspace = "LAB"; image.bpp = 24; break; default: image.colorspace = "Unknown"; image.bpp = 0; } image.compression = "JPEG2000"; if((gs.cmptlut[0] = jas_image_getcmptbytype(gs.image, JAS_IMAGE_CT_COLOR(JAS_CLRSPC_CHANIND_RGB_R))) < 0 || (gs.cmptlut[1] = jas_image_getcmptbytype(gs.image, JAS_IMAGE_CT_COLOR(JAS_CLRSPC_CHANIND_RGB_G))) < 0 || (gs.cmptlut[2] = jas_image_getcmptbytype(gs.image, JAS_IMAGE_CT_COLOR(JAS_CLRSPC_CHANIND_RGB_B))) < 0) return SQE_R_NOMEMORY; const s32 *cmptlut = gs.cmptlut; // check that all components have the same size. const s32 width = jas_image_cmptwidth(gs.image, cmptlut[0]); const s32 height = jas_image_cmptheight(gs.image, cmptlut[0]); for(s32 i = 1; i < 3; ++i) { if(jas_image_cmptwidth(gs.image, cmptlut[i]) != width || jas_image_cmptheight(gs.image, cmptlut[i]) != height) return SQE_R_BADFILE; } for(s32 i = 0;i < 3;i++) { if(!(gs.data[i] = jas_matrix_create(1, image.w))) return SQE_R_BADFILE; } finfo.image.push_back(image); line = -1; return SQE_OK; } s32 fmt_codec::read_next_pass() { return SQE_OK; } s32 fmt_codec::read_scanline(RGBA *scan) { fmt_image *im = image(currentImage); jas_seqent_t v; fmt_utils::fillAlpha(scan, im->w); line++; u8 *data = (u8 *)scan; for(s32 cmptno = 0; cmptno < 3; ++cmptno) { if(jas_image_readcmpt(gs.image, gs.cmptlut[cmptno], 0, line, im->w, 1, gs.data[cmptno])) return SQE_R_BADFILE; gs.d[cmptno] = jas_matrix_getref(gs.data[cmptno], 0, 0); } for(s32 x = 0; x < im->w;++x) { for(int k = 0; k < 3; ++k) { v = *gs.d[k]; if(v < 0) v = 0; else if(v > 255) v = 255; *data = v; data++; ++gs.d[k]; } data++; } return SQE_OK; } void fmt_codec::read_close() { if (!jasperInitialized) { return; } for(s32 cmptno = 0; cmptno < 3; ++cmptno) { if (gs.data[cmptno]) jas_matrix_destroy(gs.data[cmptno]); } if(gs.image) jas_image_destroy(gs.image); if(gs.altimage) jas_image_destroy(gs.altimage); finfo.meta.clear(); finfo.image.clear(); } // helper method bool fmt_codec::convert_colorspace() { jas_cmprof_t *outprof = jas_cmprof_createfromclrspc(JAS_CLRSPC_SRGB); if(!outprof) return false; gs.altimage = jas_image_chclrspc(gs.image, outprof, JAS_CMXFORM_INTENT_PER); if(!gs.altimage) return false; jas_cmprof_destroy(outprof); return true; } #include "fmt_codec_cd_func.h"