/*===========================================================================*\ * * * OpenMesh * * Copyright (C) 2001-2011 by Computer Graphics Group, RWTH Aachen * * www.openmesh.org * * * *---------------------------------------------------------------------------* * This file is part of OpenMesh. * * * * OpenMesh 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 3 of * * the License, or (at your option) any later version with the * * following exceptions: * * * * If other files instantiate templates or use macros * * or inline functions from this file, or you compile this file and * * link it with other files to produce an executable, this file does * * not by itself cause the resulting executable to be covered by the * * GNU Lesser General Public License. This exception does not however * * invalidate any other reasons why the executable file might be * * covered by the GNU Lesser General Public License. * * * * OpenMesh 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 LesserGeneral Public * * License along with OpenMesh. If not, * * see . * * * \*===========================================================================*/ /*===========================================================================*\ * * * $Revision$ * * $Date$ * * * \*===========================================================================*/ //============================================================================= // // Helper Functions for binary reading / writing // //============================================================================= #ifndef OPENMESH_SR_BINARY_SPEC_HH #define OPENMESH_SR_BINARY_SPEC_HH //== INCLUDES ================================================================= #include // -------------------- STL #include #include #if defined(OM_CC_GCC) && (OM_CC_VERSION < 30000) # include #else # include #endif #include #include // logic_error #include // accumulate // -------------------- OpenMesh #include #include #include #include #include //== NAMESPACES =============================================================== namespace OpenMesh { namespace IO { //============================================================================= #ifndef DOXY_IGNORE_THIS //----------------------------------------------------------------------------- // struct binary, helper for storing/restoring #define SIMPLE_BINARY( T ) \ template <> struct binary< T > { \ typedef T value_type; \ static const bool is_streamable = true; \ static size_t size_of(const value_type&) { return sizeof(value_type); } \ static size_t size_of(void) { return sizeof(value_type); } \ static size_t store( std::ostream& _os, const value_type& _val, \ bool _swap=false) { \ value_type tmp = _val; \ if (_swap) reverse_byte_order(tmp); \ _os.write( (const char*)&tmp, sizeof(value_type) ); \ return _os.good() ? sizeof(value_type) : 0; \ } \ \ static size_t restore( std::istream& _is, value_type& _val, \ bool _swap=false) { \ _is.read( (char*)&_val, sizeof(value_type) ); \ if (_swap) reverse_byte_order(_val); \ return _is.good() ? sizeof(value_type) : 0; \ } \ } SIMPLE_BINARY(bool); //SIMPLE_BINARY(int); // Why is this needed? Should not be used as not 32 bit compatible //SIMPLE_BINARY(unsigned long); SIMPLE_BINARY(float); SIMPLE_BINARY(double); SIMPLE_BINARY(long double); SIMPLE_BINARY(int8_t); SIMPLE_BINARY(int16_t); SIMPLE_BINARY(int32_t); SIMPLE_BINARY(int64_t); SIMPLE_BINARY(uint8_t); SIMPLE_BINARY(uint16_t); SIMPLE_BINARY(uint32_t); SIMPLE_BINARY(uint64_t); #undef SIMPLE_BINARY // For unsigned long which is of size 64 bit on 64 bit // architectures: convert into 32 bit unsigned integer value // in order to stay compatible between 32/64 bit architectures. // This allows cross reading BUT forbids storing unsigned longs // as data type since higher order word (4 bytes) will be truncated. // Does not work in case the data type that is to be stored // exceeds the value range of unsigned int in size, which is improbable... #define SIMPLE_BINARY( T ) \ template <> struct binary< T > { \ typedef T value_type; \ static const bool is_streamable = true; \ static size_t size_of(const value_type&) { return sizeof(value_type); } \ static size_t size_of(void) { return sizeof(value_type); } \ static size_t store( std::ostream& _os, const value_type& _val, \ bool _swap=false) { \ value_type tmp = _val; \ if (_swap) reverse_byte_order(tmp); \ /* Convert unsigned long to unsigned int for compatibility reasons */ \ unsigned int t1 = static_cast(tmp); \ _os.write( (const char*)&t1, sizeof(unsigned int) ); \ return _os.good() ? sizeof(unsigned int) : 0; \ } \ \ static size_t restore( std::istream& _is, value_type& _val, \ bool _swap=false) { \ unsigned int t1; \ _is.read( (char*)&t1, sizeof(unsigned int) ); \ _val = t1; \ if (_swap) reverse_byte_order(_val); \ return _is.good() ? sizeof(unsigned int) : 0; \ } \ } SIMPLE_BINARY(unsigned long); #undef SIMPLE_BINARY #define VECTORT_BINARY( T ) \ template <> struct binary< T > { \ typedef T value_type; \ static const bool is_streamable = true; \ static size_t size_of(void) { return sizeof(value_type); } \ static size_t size_of(const value_type&) { return size_of(); } \ static size_t store( std::ostream& _os, const value_type& _val, \ bool _swap=false) { \ value_type tmp = _val; \ size_t i, b = size_of(_val), N = value_type::size_; \ if (_swap) for (i=0; i struct binary< std::string > { typedef std::string value_type; typedef uint16_t length_t; static const bool is_streamable = true; static size_t size_of() { return UnknownSize; } static size_t size_of(const value_type &_v) { return sizeof(length_t) + _v.size(); } static size_t store(std::ostream& _os, const value_type& _v, bool _swap=false) { #if defined(OM_CC_GCC) && (OM_CC_VERSION < 30000) if (_v.size() < Utils::NumLimitsT::max() ) #else if (_v.size() < std::numeric_limits::max() ) #endif { length_t len = _v.size(); if (_swap) reverse_byte_order(len); size_t bytes = binary::store( _os, len, _swap ); _os.write( _v.data(), len ); return _os.good() ? len+bytes : 0; } throw std::runtime_error("Cannot store string longer than 64Kb"); } static size_t restore(std::istream& _is, value_type& _val, bool _swap=false) { length_t len; size_t bytes = binary::restore( _is, len, _swap ); if (_swap) reverse_byte_order(len); _val.resize(len); _is.read( const_cast(_val.data()), len ); return _is.good() ? (len+bytes) : 0; } }; template <> struct binary { typedef OpenMesh::Attributes::StatusInfo value_type; typedef value_type::value_type status_t; static const bool is_streamable = true; static size_t size_of() { return sizeof(status_t); } static size_t size_of(const value_type&) { return size_of(); } static size_t n_bytes(size_t _n_elem) { return _n_elem*sizeof(status_t); } static size_t store(std::ostream& _os, const value_type& _v, bool _swap=false) { status_t v=_v.bits(); return binary::store(_os, v, _swap); } static size_t restore( std::istream& _os, value_type& _v, bool _swap=false) { status_t v; size_t b = binary::restore(_os, v, _swap); _v.set_bits(v); return b; } }; //----------------------------------------------------------------------------- // std::vector specializations for struct binary<> template struct FunctorStore { FunctorStore( std::ostream& _os, bool _swap) : os_(_os), swap_(_swap) { } size_t operator () ( size_t _v1, const T& _s2 ) { return _v1+binary::store(os_, _s2, swap_ ); } std::ostream& os_; bool swap_; }; template struct FunctorRestore { FunctorRestore( std::istream& _is, bool _swap) : is_(_is), swap_(_swap) { } size_t operator () ( size_t _v1, T& _s2 ) { return _v1+binary::restore(is_, _s2, swap_ ); } std::istream& is_; bool swap_; }; #include #include #include // ---------------------------------------------------------------------------- #endif // DOXY_IGNORE_THIS //============================================================================= } // namespace IO } // namespace OpenMesh //============================================================================= #endif // OPENMESH_SR_BINARY_SPEC_HH defined //=============================================================================