diff --git a/Code/IO/otbJPEG2000ImageIO.cxx b/Code/IO/otbJPEG2000ImageIO.cxx index 336dd3de6923ecd0497fc8f1e0c0fd202ab3193c..6eb37da042ea65642e055fbb306c17a01b40166c 100644 --- a/Code/IO/otbJPEG2000ImageIO.cxx +++ b/Code/IO/otbJPEG2000ImageIO.cxx @@ -286,19 +286,8 @@ public: unsigned int m_ResolutionFactor; - boost::shared_ptr<opj_codestream_info_v2> GetCstrInfo() - { - return m_CstrInfo; - } - private: - boost::shared_ptr<opj_codec_t> m_Codec; boost::shared_ptr<FILE> m_File; - boost::shared_ptr<opj_image_t> m_Image; - boost::shared_ptr<opj_stream_t> m_Stream; - boost::shared_ptr<opj_codestream_info_v2> m_CstrInfo; - opj_event_mgr_t m_EventManager; - int Initialize(); }; @@ -311,6 +300,7 @@ int JPEG2000InternalReader::Open(const char *filename, unsigned int resolution) this->m_File = boost::shared_ptr<FILE>(fopen(filename, "rb"),fclose); if (!this->m_File) { + this->Clean(); return 0; } @@ -347,12 +337,7 @@ int JPEG2000InternalReader::Open(const char *filename, unsigned int resolution) void JPEG2000InternalReader::Clean() { - // Clear opj members and file pointer - m_Image = boost::shared_ptr<opj_image_t>(); - m_Stream = boost::shared_ptr<opj_stream_t>(); m_File = boost::shared_ptr<FILE>(); - m_Codec = boost::shared_ptr<opj_codec_t>(); - m_CstrInfo = boost::shared_ptr<opj_codestream_info_v2>(); this->m_XResolution.clear(); this->m_YResolution.clear(); @@ -374,12 +359,64 @@ void JPEG2000InternalReader::Clean() boost::shared_ptr<opj_image_t> JPEG2000InternalReader::DecodeTile(unsigned int tileIndex) { - boost::shared_ptr<opj_image_t> image (otbopenjpeg_opj_image_create0(),OpjImageDestroy); - otbopenjpeg_opj_copy_image_header(m_Image.get(), image.get()); - bool success = otbopenjpeg_opj_get_decoded_tile(m_Codec.get(), m_Stream.get(), image.get(), tileIndex); + if (!m_File) + { + this->Clean(); + return boost::shared_ptr<opj_image_t>(); + } + + // Creating the file stream + boost::shared_ptr<opj_stream_t> stream = boost::shared_ptr<opj_stream_t>(otbopenjpeg_opj_stream_create_default_file_stream(m_File.get(), true),otbopenjpeg_opj_stream_destroy); + if (!stream) + { + this->Clean(); + return boost::shared_ptr<opj_image_t>(); + } + + // Creating the codec + boost::shared_ptr<opj_codec_t> codec = boost::shared_ptr<opj_codec_t>(otbopenjpeg_opj_create_decompress_v2(this->m_CodecFormat),otbopenjpeg_opj_destroy_codec); + + if (!codec) + { + this->Clean(); + return boost::shared_ptr<opj_image_t>(); + } + + // Setting default parameters + opj_dparameters_t parameters; + otbopenjpeg_opj_set_default_decoder_parameters(¶meters); + parameters.cp_reduce = static_cast<int>(this->m_ResolutionFactor); + + otbMsgDevMacro( << "Initialize decoder with cp_reduce = " << parameters.cp_reduce); + + // Set default event mgr + opj_event_mgr_t eventManager; + eventManager.info_handler = info_callback; + eventManager.warning_handler = warning_callback; + eventManager.error_handler = error_callback; + + // Setup the decoder decoding parameters using user parameters + if (!otbopenjpeg_opj_setup_decoder_v2(codec.get(), ¶meters, &eventManager)) + { + this->Clean(); + return boost::shared_ptr<opj_image_t>(); + } + + // Read the main header of the codestream and if necessary the JP2 + // boxes + opj_image_t * unsafeOpjImgPtr = NULL; + + if (!otbopenjpeg_opj_read_header(stream.get(), codec.get(),&unsafeOpjImgPtr)) + { + boost::shared_ptr<opj_image_t> image = boost::shared_ptr<opj_image_t>(unsafeOpjImgPtr,OpjImageDestroy); + this->Clean(); + return boost::shared_ptr<opj_image_t>(); + } + + boost::shared_ptr<opj_image_t> image = boost::shared_ptr<opj_image_t>(unsafeOpjImgPtr,OpjImageDestroy); - if(success) + if(otbopenjpeg_opj_get_decoded_tile(codec.get(), stream.get(), image.get(), tileIndex)) { otbMsgDevMacro(<<"Tile "<<tileIndex<<" read from file"); return image; @@ -392,12 +429,6 @@ boost::shared_ptr<opj_image_t> JPEG2000InternalReader::DecodeTile(unsigned int t JPEG2000InternalReader::JPEG2000InternalReader() { - // Set default event mgr - memset(&m_EventManager, 0, sizeof(opj_event_mgr_t)); - m_EventManager.info_handler = info_callback; - m_EventManager.warning_handler = warning_callback; - m_EventManager.error_handler = error_callback; - this->Clean(); } @@ -406,17 +437,17 @@ int JPEG2000InternalReader::Initialize() if (m_File) { // Creating the file stream - m_Stream = boost::shared_ptr<opj_stream_t>(otbopenjpeg_opj_stream_create_default_file_stream(m_File.get(), true),otbopenjpeg_opj_stream_destroy); - if (!m_Stream) + boost::shared_ptr<opj_stream_t> stream = boost::shared_ptr<opj_stream_t>(otbopenjpeg_opj_stream_create_default_file_stream(m_File.get(), true),otbopenjpeg_opj_stream_destroy); + if (!stream) { this->Clean(); return 0; } // Creating the codec - m_Codec = boost::shared_ptr<opj_codec_t>(otbopenjpeg_opj_create_decompress_v2(this->m_CodecFormat),otbopenjpeg_opj_destroy_codec); + boost::shared_ptr<opj_codec_t> codec = boost::shared_ptr<opj_codec_t>(otbopenjpeg_opj_create_decompress_v2(this->m_CodecFormat),otbopenjpeg_opj_destroy_codec); - if (!m_Codec) + if (!codec) { this->Clean(); return 0; @@ -429,8 +460,14 @@ int JPEG2000InternalReader::Initialize() otbMsgDevMacro( << "Initialize decoder with cp_reduce = " << parameters.cp_reduce); + // Set default event mgr + opj_event_mgr_t eventManager; + eventManager.info_handler = info_callback; + eventManager.warning_handler = warning_callback; + eventManager.error_handler = error_callback; + // Setup the decoder decoding parameters using user parameters - if (!otbopenjpeg_opj_setup_decoder_v2(m_Codec.get(), ¶meters, &m_EventManager)) + if (!otbopenjpeg_opj_setup_decoder_v2(codec.get(), ¶meters, &eventManager)) { this->Clean(); return 0; @@ -439,19 +476,20 @@ int JPEG2000InternalReader::Initialize() // Read the main header of the codestream and if necessary the JP2 // boxes opj_image_t * unsafeOpjImgPtr = NULL; - if (!otbopenjpeg_opj_read_header(m_Stream.get(), m_Codec.get(),&unsafeOpjImgPtr)) + + if (!otbopenjpeg_opj_read_header(stream.get(), codec.get(),&unsafeOpjImgPtr)) { - m_Image = boost::shared_ptr<opj_image_t>(unsafeOpjImgPtr,OpjImageDestroy); + boost::shared_ptr<opj_image_t> image = boost::shared_ptr<opj_image_t>(unsafeOpjImgPtr,OpjImageDestroy); this->Clean(); return 0; } - m_Image = boost::shared_ptr<opj_image_t>(unsafeOpjImgPtr,OpjImageDestroy); + boost::shared_ptr<opj_image_t> image = boost::shared_ptr<opj_image_t>(unsafeOpjImgPtr,OpjImageDestroy); // Get the codestream information - this->m_CstrInfo = boost::shared_ptr<opj_codestream_info_v2>(otbopenjpeg_opj_get_cstr_info(m_Codec.get()),OpjCodestreamDestroy); + boost::shared_ptr<opj_codestream_info_v2> cstrInfo = boost::shared_ptr<opj_codestream_info_v2>(otbopenjpeg_opj_get_cstr_info(codec.get()),OpjCodestreamDestroy); - if (!m_CstrInfo) + if (!cstrInfo) { std::cerr << "ERROR while get codestream info" << std::endl; this->Clean(); @@ -460,33 +498,33 @@ int JPEG2000InternalReader::Initialize() // We can now retrieve the main information of the image and the codestream // (based on the first component and with no subsampling) - this->m_Width = this->m_Image->comps->w; - this->m_Height = this->m_Image->comps->h; + this->m_Width = image->comps->w; + this->m_Height = image->comps->h; otbMsgDevMacro(<< "JPEG2000InternalReader dimension (after reading header) = " << this->m_Image->comps->w << " x " - << this->m_Image->comps->h ); + << image->comps->h ); - this->m_TileHeight = this->m_CstrInfo->tdy; - this->m_TileWidth = this->m_CstrInfo->tdx; - this->m_XNbOfTile = this->m_CstrInfo->tw; - this->m_YNbOfTile = this->m_CstrInfo->th; + this->m_TileHeight = cstrInfo->tdy; + this->m_TileWidth = cstrInfo->tdx; + this->m_XNbOfTile = cstrInfo->tw; + this->m_YNbOfTile = cstrInfo->th; - this->m_NbOfComponent = this->m_Image->numcomps; + this->m_NbOfComponent = image->numcomps; for (unsigned int itComp = 0; itComp < this->m_NbOfComponent; itComp++) { - this->m_Precision.push_back( this->m_Image->comps[itComp].prec); - this->m_Signed.push_back( this->m_Image->comps[itComp].sgnd); - this->m_XResolution.push_back( this->m_Image->comps[itComp].dx); - this->m_YResolution.push_back( this->m_Image->comps[itComp].dy); + this->m_Precision.push_back( image->comps[itComp].prec); + this->m_Signed.push_back( image->comps[itComp].sgnd); + this->m_XResolution.push_back( image->comps[itComp].dx); + this->m_YResolution.push_back( image->comps[itComp].dy); } - } - // Warning: This value is based on the first component of the default tile parameters. - unsigned int numResAvailable = this->m_CstrInfo->m_default_tile_info.tccp_info[0].numresolutions; - for (unsigned int itRes = 0; itRes < numResAvailable; itRes++) - { - m_AvailableResolutions.push_back(itRes); + // Warning: This value is based on the first component of the default tile parameters. + unsigned int numResAvailable = cstrInfo->m_default_tile_info.tccp_info[0].numresolutions; + for (unsigned int itRes = 0; itRes < numResAvailable; itRes++) + { + m_AvailableResolutions.push_back(itRes); + } } return 1; @@ -495,11 +533,7 @@ int JPEG2000InternalReader::Initialize() int JPEG2000InternalReader::CanRead() { - if ( this->m_File && - this->m_Codec && - this->m_Stream && - this->m_CstrInfo && - this->m_Image && + if (Initialize() && ( this->m_Width > 0 ) && ( this->m_Height > 0 ) && ( this->m_TileWidth > 0 ) && ( this->m_TileHeight > 0 ) && ( this->m_XNbOfTile > 0 ) && ( this->m_YNbOfTile > 0 ) && @@ -795,8 +829,8 @@ bool JPEG2000ImageIO::GetResolutionInfo(std::vector<unsigned int>& res, std::vec int w = int_ceildivpow2( originalWidth, *itRes); int h = int_ceildivpow2( originalHeight, *itRes); - int tw = int_ceildivpow2(m_InternalReaders[0]->GetCstrInfo()->tdx, *itRes); - int th = int_ceildivpow2(m_InternalReaders[0]->GetCstrInfo()->tdy, *itRes); + int tw = int_ceildivpow2(m_InternalReaders[0]->m_TileWidth, *itRes); + int th = int_ceildivpow2(m_InternalReaders[0]->m_TileHeight, *itRes); oss << "Resolution: " << *itRes << " (Image [w x h]: " << w << "x" << h << ", Tile [w x h]: " << tw << "x" << th << ")"; @@ -1496,8 +1530,8 @@ std::vector<unsigned int> JPEG2000ImageIO::ComputeTileList() unsigned int tile_size_y = m_InternalReaders.front()->m_TileHeight; unsigned int width = m_Dimensions[0]; unsigned int height = m_Dimensions[1]; - unsigned int nbOfTileX = m_InternalReaders.front()->GetCstrInfo()->tw; - unsigned int nbOfTileY = m_InternalReaders.front()->GetCstrInfo()->th; + unsigned int nbOfTileX = m_InternalReaders.front()->m_XNbOfTile; + unsigned int nbOfTileY = m_InternalReaders.front()->m_YNbOfTile; unsigned int tilePosX0, tilePosX1; unsigned int tilePosY0, tilePosY1;