// This file is part of OpenCV project. // It is subject to the license terms in the LICENSE file found in the top-level directory // of this distribution and at http://opencv.org/license.html. #include "precomp.hpp" #include "utils.hpp" #include "grfmt_pfm.hpp" #include #ifdef HAVE_IMGCODEC_PFM namespace { static_assert(sizeof(float) == 4, "float must be 32 bit."); bool is_byte_order_swapped(double scale) { // ".pfm" format file specifies that: // positive scale means big endianness; // negative scale means little endianness. #ifdef WORDS_BIGENDIAN return scale < 0.0; #else return scale >= 0.0; #endif } void swap_endianess(uint32_t& ui) { static const uint32_t A(0x000000ffU); static const uint32_t B(0x0000ff00U); static const uint32_t C(0x00ff0000U); static const uint32_t D(0xff000000U); ui = ( (ui & A) << 24 ) | ( (ui & B) << 8 ) | ( (ui & C) >> 8 ) | ( (ui & D) >> 24 ); } template T atoT(const std::string& s); template<> int atoT(const std::string& s) { return std::atoi(s.c_str()); } template<> double atoT(const std::string& s) { return std::atof(s.c_str()); } template T read_number(cv::RLByteStream& strm) { // should be enough to take string representation of any number const size_t buffer_size = 2048; std::vector buffer(buffer_size, 0); for (size_t i = 0; i < buffer_size; ++i) { const int intc = strm.getByte(); CV_Assert(intc >= -128 && intc < 128); char c = static_cast(intc); if (std::isspace(c)) { break; } buffer[i] = c; } const std::string str(buffer.begin(), buffer.end()); return atoT(str); } template void write_anything(cv::WLByteStream& strm, const T& t) { std::ostringstream ss; ss << t; strm.putBytes(ss.str().c_str(), static_cast(ss.str().size())); } } namespace cv { PFMDecoder::~PFMDecoder() { } PFMDecoder::PFMDecoder() : m_scale_factor(0), m_swap_byte_order(false) { m_strm.close(); } bool PFMDecoder::readHeader() { if (m_buf.empty()) { if (!m_strm.open(m_filename)) { return false; } } else { if (!m_strm.open(m_buf)) { return false; } } if (m_strm.getByte() != 'P') { CV_Error(Error::StsError, "Unexpected file type (expected P)"); } switch (m_strm.getByte()) { case 'f': m_type = CV_32FC1; break; case 'F': m_type = CV_32FC3; break; default: CV_Error(Error::StsError, "Unexpected file type (expected `f` or `F`)"); } if ('\n' != m_strm.getByte()) { CV_Error(Error::StsError, "Unexpected header format (expected line break)"); } m_width = read_number(m_strm); m_height = read_number(m_strm); m_scale_factor = read_number(m_strm); m_swap_byte_order = is_byte_order_swapped(m_scale_factor); return true; } bool PFMDecoder::readData(Mat& mat) { if (!m_strm.isOpened()) { CV_Error(Error::StsError, "Unexpected status in data stream"); } Mat buffer(mat.size(), m_type); for (int y = m_height - 1; y >= 0; --y) { m_strm.getBytes(buffer.ptr(y), static_cast(m_width * buffer.elemSize())); if (is_byte_order_swapped(m_scale_factor)) { for (int i = 0; i < m_width * buffer.channels(); ++i) { static_assert( sizeof(uint32_t) == sizeof(float), "uint32_t and float must have same size." ); swap_endianess(buffer.ptr(y)[i]); } } } if (buffer.channels() == 3) { cv::cvtColor(buffer, buffer, cv::COLOR_BGR2RGB); } CV_Assert(fabs(m_scale_factor) > 0.0f); buffer *= 1.f / fabs(m_scale_factor); buffer.convertTo(mat, mat.type()); return true; } size_t PFMDecoder::signatureLength() const { return 3; } bool PFMDecoder::checkSignature( const String& signature ) const { return signature.size() >= 3 && signature[0] == 'P' && ( signature[1] == 'f' || signature[1] == 'F' ) && isspace(signature[2]); } void PFMDecoder::close() { // noop } ////////////////////////////////////////////////////////////////////////////////////////// PFMEncoder::PFMEncoder() { m_description = "Portable image format - float (*.pfm)"; } PFMEncoder::~PFMEncoder() { } bool PFMEncoder::isFormatSupported(int depth) const { // any depth will be converted into 32-bit float. CV_UNUSED(depth); return true; } bool PFMEncoder::write(const Mat& img, const std::vector& params) { CV_UNUSED(params); WLByteStream strm; if (m_buf) { if (!strm.open(*m_buf)) { return false; } else { m_buf->reserve(alignSize(256 + sizeof(float) * img.channels() * img.total(), 256)); } } else if (!strm.open(m_filename)) { return false; } Mat float_img; strm.putByte('P'); switch (img.channels()) { case 1: strm.putByte('f'); img.convertTo(float_img, CV_32FC1); break; case 3: strm.putByte('F'); img.convertTo(float_img, CV_32FC3); break; default: CV_Error(Error::StsBadArg, "Expected 1 or 3 channel image."); } strm.putByte('\n'); write_anything(strm, float_img.cols); strm.putByte(' '); write_anything(strm, float_img.rows); strm.putByte('\n'); #ifdef WORDS_BIGENDIAN write_anything(strm, 1.0); #else write_anything(strm, -1.0); #endif strm.putByte('\n'); // Comments are not officially supported in this file format. // write_anything(strm, "# Generated by OpenCV " CV_VERSION "\n"); for (int y = float_img.rows - 1; y >= 0; --y) { if (float_img.channels() == 3) { const float* bgr_row = float_img.ptr(y); size_t row_size = float_img.cols * float_img.channels(); std::vector rgb_row(row_size); for (int x = 0; x < float_img.cols; ++x) { rgb_row[x*3+0] = bgr_row[x*3+2]; rgb_row[x*3+1] = bgr_row[x*3+1]; rgb_row[x*3+2] = bgr_row[x*3+0]; } strm.putBytes( reinterpret_cast(rgb_row.data()), static_cast(sizeof(float) * row_size) ); } else if (float_img.channels() == 1) { strm.putBytes(float_img.ptr(y), sizeof(float) * float_img.cols); } } return true; } } #endif // HAVE_IMGCODEC_PFM