mirror of
https://github.com/boostorg/endian.git
synced 2025-07-31 13:07:24 +02:00
Initial commit buffers.hpp and related infrastructure. A separate buffer class without arithmetic operators was requested during the formal review.
This commit is contained in:
602
include/boost/endian/buffers.hpp
Normal file
602
include/boost/endian/buffers.hpp
Normal file
@@ -0,0 +1,602 @@
|
|||||||
|
// boost/endian/types.hpp ------------------------------------------------------------//
|
||||||
|
|
||||||
|
// (C) Copyright Darin Adler 2000
|
||||||
|
// (C) Copyright Beman Dawes 2006, 2009
|
||||||
|
|
||||||
|
// Distributed under the Boost Software License, Version 1.0.
|
||||||
|
// See http://www.boost.org/LICENSE_1_0.txt
|
||||||
|
|
||||||
|
// See library home page at http://www.boost.org/libs/endian
|
||||||
|
|
||||||
|
//--------------------------------------------------------------------------------------//
|
||||||
|
|
||||||
|
// Original design developed by Darin Adler based on classes developed by Mark
|
||||||
|
// Borgerding. Four original class templates were combined into a single endian
|
||||||
|
// class template by Beman Dawes, who also added the unrolled_byte_loops sign
|
||||||
|
// partial specialization to correctly extend the sign when cover integer size
|
||||||
|
// differs from endian representation size.
|
||||||
|
|
||||||
|
// TODO: When a compiler supporting constexpr becomes available, try possible uses.
|
||||||
|
|
||||||
|
#ifndef BOOST_ENDIAN_BUFFERS_HPP
|
||||||
|
#define BOOST_ENDIAN_BUFFERS_HPP
|
||||||
|
|
||||||
|
#if defined(_MSC_VER)
|
||||||
|
# pragma warning(push)
|
||||||
|
# pragma warning(disable:4365) // conversion ... signed/unsigned mismatch
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef BOOST_ENDIAN_LOG
|
||||||
|
# include <iostream>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if defined(__BORLANDC__) || defined( __CODEGEARC__)
|
||||||
|
# pragma pack(push, 1)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include <boost/config.hpp>
|
||||||
|
#include <boost/predef/detail/endian_compat.h>
|
||||||
|
#include <boost/endian/conversion.hpp>
|
||||||
|
#define BOOST_MINIMAL_INTEGER_COVER_OPERATORS
|
||||||
|
#include <boost/endian/detail/cover_operators.hpp>
|
||||||
|
#undef BOOST_MINIMAL_INTEGER_COVER_OPERATORS
|
||||||
|
#include <boost/type_traits/is_signed.hpp>
|
||||||
|
#include <boost/cstdint.hpp>
|
||||||
|
#include <boost/static_assert.hpp>
|
||||||
|
#include <boost/detail/scoped_enum_emulation.hpp>
|
||||||
|
#include <iosfwd>
|
||||||
|
#include <climits>
|
||||||
|
|
||||||
|
# if CHAR_BIT != 8
|
||||||
|
# error Platforms with CHAR_BIT != 8 are not supported
|
||||||
|
# endif
|
||||||
|
|
||||||
|
# ifdef BOOST_NO_CXX11_DEFAULTED_FUNCTIONS
|
||||||
|
# define BOOST_ENDIAN_DEFAULT_CONSTRUCT {} // C++03
|
||||||
|
# else
|
||||||
|
# define BOOST_ENDIAN_DEFAULT_CONSTRUCT = default; // C++0x
|
||||||
|
# endif
|
||||||
|
|
||||||
|
# if defined(BOOST_NO_CXX11_DEFAULTED_FUNCTIONS) && defined(BOOST_ENDIAN_FORCE_PODNESS)
|
||||||
|
# define BOOST_ENDIAN_NO_CTORS
|
||||||
|
# endif
|
||||||
|
|
||||||
|
# ifndef BOOST_ENDIAN_EXPLICIT_CTORS
|
||||||
|
# define BOOST_ENDIAN_EXPLICIT_OPT
|
||||||
|
# else
|
||||||
|
# define BOOST_ENDIAN_EXPLICIT_OPT explicit
|
||||||
|
# endif
|
||||||
|
|
||||||
|
//---------------------------------- synopsis ----------------------------------------//
|
||||||
|
|
||||||
|
namespace boost
|
||||||
|
{
|
||||||
|
namespace endian
|
||||||
|
{
|
||||||
|
|
||||||
|
#ifndef BOOST_ENDIAN_ORDER_ENUM_DEFINED
|
||||||
|
BOOST_SCOPED_ENUM_START(order)
|
||||||
|
{
|
||||||
|
big, little,
|
||||||
|
# ifdef BOOST_BIG_ENDIAN
|
||||||
|
native = big
|
||||||
|
# else
|
||||||
|
native = little
|
||||||
|
# endif
|
||||||
|
}; BOOST_SCOPED_ENUM_END
|
||||||
|
# define BOOST_ENDIAN_ORDER_ENUM_DEFINED
|
||||||
|
#endif
|
||||||
|
BOOST_SCOPED_ENUM_START(align) {no, yes}; BOOST_SCOPED_ENUM_END
|
||||||
|
|
||||||
|
template <BOOST_SCOPED_ENUM(order) Order, class T, std::size_t n_bits,
|
||||||
|
BOOST_SCOPED_ENUM(align) A = align::no>
|
||||||
|
class endian_buffer;
|
||||||
|
|
||||||
|
// aligned big endian floating point types
|
||||||
|
typedef endian_buffer<order::big, float, 32, align::yes> big_align_floatbuf32_t;
|
||||||
|
typedef endian_buffer<order::big, double, 64, align::yes> big_align_floatbuf64_t;
|
||||||
|
|
||||||
|
// aligned little endian floating point types
|
||||||
|
typedef endian_buffer<order::little, float, 32, align::yes> little_align_floatbuf32_t;
|
||||||
|
typedef endian_buffer<order::little, double, 64, align::yes> little_align_floatbuf64_t;
|
||||||
|
|
||||||
|
// unaligned big endian floating point types
|
||||||
|
typedef endian_buffer<order::big, float, 32, align::no> big_floatbuf32_t;
|
||||||
|
typedef endian_buffer<order::big, double, 64, align::no> big_floatbuf64_t;
|
||||||
|
|
||||||
|
// unaligned little endian floating point types
|
||||||
|
typedef endian_buffer<order::little, float, 32, align::no> little_floatbuf32_t;
|
||||||
|
typedef endian_buffer<order::little, double, 64, align::no> little_floatbuf64_t;
|
||||||
|
|
||||||
|
// aligned big endian signed integer types
|
||||||
|
typedef endian_buffer<order::big, int16_t, 16, align::yes> big_align_buf16_t;
|
||||||
|
typedef endian_buffer<order::big, int32_t, 32, align::yes> big_align_buf32_t;
|
||||||
|
typedef endian_buffer<order::big, int64_t, 64, align::yes> big_align_buf64_t;
|
||||||
|
|
||||||
|
// aligned big endian unsigned integer types
|
||||||
|
typedef endian_buffer<order::big, uint16_t, 16, align::yes> big_align_ubuf16_t;
|
||||||
|
typedef endian_buffer<order::big, uint32_t, 32, align::yes> big_align_ubuf32_t;
|
||||||
|
typedef endian_buffer<order::big, uint64_t, 64, align::yes> big_align_ubuf64_t;
|
||||||
|
|
||||||
|
// aligned little endian signed integer types
|
||||||
|
typedef endian_buffer<order::little, int16_t, 16, align::yes> little_align_buf16_t;
|
||||||
|
typedef endian_buffer<order::little, int32_t, 32, align::yes> little_align_buf32_t;
|
||||||
|
typedef endian_buffer<order::little, int64_t, 64, align::yes> little_align_buf64_t;
|
||||||
|
|
||||||
|
// aligned little endian unsigned integer types
|
||||||
|
typedef endian_buffer<order::little, uint16_t, 16, align::yes> little_align_ubuf16_t;
|
||||||
|
typedef endian_buffer<order::little, uint32_t, 32, align::yes> little_align_ubuf32_t;
|
||||||
|
typedef endian_buffer<order::little, uint64_t, 64, align::yes> little_align_ubuf64_t;
|
||||||
|
|
||||||
|
// aligned native endian typedefs are not provided because
|
||||||
|
// <cstdint> types are superior for this use case
|
||||||
|
|
||||||
|
// unaligned big endian signed integer types
|
||||||
|
typedef endian_buffer<order::big, int_least8_t, 8> big_buf8_t;
|
||||||
|
typedef endian_buffer<order::big, int_least16_t, 16> big_buf16_t;
|
||||||
|
typedef endian_buffer<order::big, int_least32_t, 24> big_buf24_t;
|
||||||
|
typedef endian_buffer<order::big, int_least32_t, 32> big_buf32_t;
|
||||||
|
typedef endian_buffer<order::big, int_least64_t, 40> big_buf40_t;
|
||||||
|
typedef endian_buffer<order::big, int_least64_t, 48> big_buf48_t;
|
||||||
|
typedef endian_buffer<order::big, int_least64_t, 56> big_buf56_t;
|
||||||
|
typedef endian_buffer<order::big, int_least64_t, 64> big_buf64_t;
|
||||||
|
|
||||||
|
// unaligned big endian unsigned integer types
|
||||||
|
typedef endian_buffer<order::big, uint_least8_t, 8> big_ubuf8_t;
|
||||||
|
typedef endian_buffer<order::big, uint_least16_t, 16> big_ubuf16_t;
|
||||||
|
typedef endian_buffer<order::big, uint_least32_t, 24> big_ubuf24_t;
|
||||||
|
typedef endian_buffer<order::big, uint_least32_t, 32> big_ubuf32_t;
|
||||||
|
typedef endian_buffer<order::big, uint_least64_t, 40> big_ubuf40_t;
|
||||||
|
typedef endian_buffer<order::big, uint_least64_t, 48> big_ubuf48_t;
|
||||||
|
typedef endian_buffer<order::big, uint_least64_t, 56> big_ubuf56_t;
|
||||||
|
typedef endian_buffer<order::big, uint_least64_t, 64> big_ubuf64_t;
|
||||||
|
|
||||||
|
// unaligned little endian signed integer types
|
||||||
|
typedef endian_buffer<order::little, int_least8_t, 8> little_buf8_t;
|
||||||
|
typedef endian_buffer<order::little, int_least16_t, 16> little_buf16_t;
|
||||||
|
typedef endian_buffer<order::little, int_least32_t, 24> little_buf24_t;
|
||||||
|
typedef endian_buffer<order::little, int_least32_t, 32> little_buf32_t;
|
||||||
|
typedef endian_buffer<order::little, int_least64_t, 40> little_buf40_t;
|
||||||
|
typedef endian_buffer<order::little, int_least64_t, 48> little_buf48_t;
|
||||||
|
typedef endian_buffer<order::little, int_least64_t, 56> little_buf56_t;
|
||||||
|
typedef endian_buffer<order::little, int_least64_t, 64> little_buf64_t;
|
||||||
|
|
||||||
|
// unaligned little endian unsigned integer types
|
||||||
|
typedef endian_buffer<order::little, uint_least8_t, 8> little_ubuf8_t;
|
||||||
|
typedef endian_buffer<order::little, uint_least16_t, 16> little_ubuf16_t;
|
||||||
|
typedef endian_buffer<order::little, uint_least32_t, 24> little_ubuf24_t;
|
||||||
|
typedef endian_buffer<order::little, uint_least32_t, 32> little_ubuf32_t;
|
||||||
|
typedef endian_buffer<order::little, uint_least64_t, 40> little_ubuf40_t;
|
||||||
|
typedef endian_buffer<order::little, uint_least64_t, 48> little_ubuf48_t;
|
||||||
|
typedef endian_buffer<order::little, uint_least64_t, 56> little_ubuf56_t;
|
||||||
|
typedef endian_buffer<order::little, uint_least64_t, 64> little_ubuf64_t;
|
||||||
|
|
||||||
|
# ifdef BOOST_BIG_ENDIAN
|
||||||
|
// unaligned native endian signed integer types
|
||||||
|
typedef big_buf8_t native_buf8_t;
|
||||||
|
typedef big_buf16_t native_buf16_t;
|
||||||
|
typedef big_buf24_t native_buf24_t;
|
||||||
|
typedef big_buf32_t native_buf32_t;
|
||||||
|
typedef big_buf40_t native_buf40_t;
|
||||||
|
typedef big_buf48_t native_buf48_t;
|
||||||
|
typedef big_buf56_t native_buf56_t;
|
||||||
|
typedef big_buf64_t native_buf64_t;
|
||||||
|
|
||||||
|
// unaligned native endian unsigned integer types
|
||||||
|
typedef big_ubuf8_t native_ubuf8_t;
|
||||||
|
typedef big_ubuf16_t native_ubuf16_t;
|
||||||
|
typedef big_ubuf24_t native_ubuf24_t;
|
||||||
|
typedef big_ubuf32_t native_ubuf32_t;
|
||||||
|
typedef big_ubuf40_t native_ubuf40_t;
|
||||||
|
typedef big_ubuf48_t native_ubuf48_t;
|
||||||
|
typedef big_ubuf56_t native_ubuf56_t;
|
||||||
|
typedef big_ubuf64_t native_ubuf64_t;
|
||||||
|
|
||||||
|
// native endian floating point types
|
||||||
|
typedef big_floatbuf32_t native_floatbuf32_t;
|
||||||
|
typedef big_floatbuf64_t native_floatbuf64_t;
|
||||||
|
typedef big_align_floatbuf32_t native_align_floatbuf32_t;
|
||||||
|
typedef big_align_floatbuf64_t native_align_floatbuf64_t;
|
||||||
|
# else
|
||||||
|
// unaligned native endian signed integer types
|
||||||
|
typedef little_buf8_t native_buf8_t;
|
||||||
|
typedef little_buf16_t native_buf16_t;
|
||||||
|
typedef little_buf24_t native_buf24_t;
|
||||||
|
typedef little_buf32_t native_buf32_t;
|
||||||
|
typedef little_buf40_t native_buf40_t;
|
||||||
|
typedef little_buf48_t native_buf48_t;
|
||||||
|
typedef little_buf56_t native_buf56_t;
|
||||||
|
typedef little_buf64_t native_buf64_t;
|
||||||
|
|
||||||
|
// unaligned native endian unsigned integer types
|
||||||
|
typedef little_ubuf8_t native_ubuf8_t;
|
||||||
|
typedef little_ubuf16_t native_ubuf16_t;
|
||||||
|
typedef little_ubuf24_t native_ubuf24_t;
|
||||||
|
typedef little_ubuf32_t native_ubuf32_t;
|
||||||
|
typedef little_ubuf40_t native_ubuf40_t;
|
||||||
|
typedef little_ubuf48_t native_ubuf48_t;
|
||||||
|
typedef little_ubuf56_t native_ubuf56_t;
|
||||||
|
typedef little_ubuf64_t native_ubuf64_t;
|
||||||
|
|
||||||
|
// native endian floating point types
|
||||||
|
typedef little_floatbuf32_t native_floatbuf32_t;
|
||||||
|
typedef little_floatbuf64_t native_floatbuf64_t;
|
||||||
|
typedef little_align_floatbuf32_t native_align_floatbuf32_t;
|
||||||
|
typedef little_align_floatbuf64_t native_align_floatbuf64_t;
|
||||||
|
# endif
|
||||||
|
} // namespace boost
|
||||||
|
} // namespace endian
|
||||||
|
|
||||||
|
//---------------------------------- end synopsis ------------------------------------//
|
||||||
|
|
||||||
|
namespace boost
|
||||||
|
{
|
||||||
|
namespace endian
|
||||||
|
{
|
||||||
|
namespace detail
|
||||||
|
{
|
||||||
|
|
||||||
|
// Unrolled loops for loading and storing streams of bytes.
|
||||||
|
|
||||||
|
template <typename T, std::size_t n_bytes,
|
||||||
|
bool sign=boost::is_signed<T>::value >
|
||||||
|
struct unrolled_byte_loops
|
||||||
|
{
|
||||||
|
typedef unrolled_byte_loops<T, n_bytes - 1, sign> next;
|
||||||
|
|
||||||
|
static T load_big(const unsigned char* bytes) BOOST_NOEXCEPT
|
||||||
|
{ return *(bytes - 1) | (next::load_big(bytes - 1) << 8); }
|
||||||
|
static T load_little(const unsigned char* bytes) BOOST_NOEXCEPT
|
||||||
|
{ return *bytes | (next::load_little(bytes + 1) << 8); }
|
||||||
|
|
||||||
|
static void store_big(char* bytes, T value) BOOST_NOEXCEPT
|
||||||
|
{
|
||||||
|
*(bytes - 1) = static_cast<char>(value);
|
||||||
|
next::store_big(bytes - 1, value >> 8);
|
||||||
|
}
|
||||||
|
static void store_little(char* bytes, T value) BOOST_NOEXCEPT
|
||||||
|
{
|
||||||
|
*bytes = static_cast<char>(value);
|
||||||
|
next::store_little(bytes + 1, value >> 8);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
struct unrolled_byte_loops<T, 1, false>
|
||||||
|
{
|
||||||
|
static T load_big(const unsigned char* bytes) BOOST_NOEXCEPT
|
||||||
|
{ return *(bytes - 1); }
|
||||||
|
static T load_little(const unsigned char* bytes) BOOST_NOEXCEPT
|
||||||
|
{ return *bytes; }
|
||||||
|
static void store_big(char* bytes, T value) BOOST_NOEXCEPT
|
||||||
|
{ *(bytes - 1) = static_cast<char>(value); }
|
||||||
|
static void store_little(char* bytes, T value) BOOST_NOEXCEPT
|
||||||
|
{ *bytes = static_cast<char>(value); }
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
struct unrolled_byte_loops<T, 1, true>
|
||||||
|
{
|
||||||
|
static T load_big(const unsigned char* bytes) BOOST_NOEXCEPT
|
||||||
|
{ return *reinterpret_cast<const signed char*>(bytes - 1); }
|
||||||
|
static T load_little(const unsigned char* bytes) BOOST_NOEXCEPT
|
||||||
|
{ return *reinterpret_cast<const signed char*>(bytes); }
|
||||||
|
static void store_big(char* bytes, T value) BOOST_NOEXCEPT
|
||||||
|
{ *(bytes - 1) = static_cast<char>(value); }
|
||||||
|
static void store_little(char* bytes, T value) BOOST_NOEXCEPT
|
||||||
|
{ *bytes = static_cast<char>(value); }
|
||||||
|
};
|
||||||
|
|
||||||
|
template <typename T, std::size_t n_bytes>
|
||||||
|
inline
|
||||||
|
T load_big_endian(const void* bytes) BOOST_NOEXCEPT
|
||||||
|
{
|
||||||
|
return unrolled_byte_loops<T, n_bytes>::load_big
|
||||||
|
(static_cast<const unsigned char*>(bytes) + n_bytes);
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename T, std::size_t n_bytes>
|
||||||
|
inline
|
||||||
|
T load_little_endian(const void* bytes) BOOST_NOEXCEPT
|
||||||
|
{
|
||||||
|
return unrolled_byte_loops<T, n_bytes>::load_little
|
||||||
|
(static_cast<const unsigned char*>(bytes));
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename T, std::size_t n_bytes>
|
||||||
|
inline
|
||||||
|
void store_big_endian(void* bytes, T value) BOOST_NOEXCEPT
|
||||||
|
{
|
||||||
|
unrolled_byte_loops<T, n_bytes>::store_big
|
||||||
|
(static_cast<char*>(bytes) + n_bytes, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename T, std::size_t n_bytes>
|
||||||
|
inline
|
||||||
|
void store_little_endian(void* bytes, T value) BOOST_NOEXCEPT
|
||||||
|
{
|
||||||
|
unrolled_byte_loops<T, n_bytes>::store_little
|
||||||
|
(static_cast<char*>(bytes), value);
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace detail
|
||||||
|
|
||||||
|
# ifdef BOOST_ENDIAN_LOG
|
||||||
|
bool endian_log(true);
|
||||||
|
# endif
|
||||||
|
|
||||||
|
// endian_buffer class template specializations --------------------------------------//
|
||||||
|
|
||||||
|
// Specializations that represent unaligned bytes.
|
||||||
|
// Taking an integer type as a parameter provides a nice way to pass both
|
||||||
|
// the size and signedness of the desired integer and get the appropriate
|
||||||
|
// corresponding integer type for the interface.
|
||||||
|
|
||||||
|
// unaligned big endian_buffer specialization
|
||||||
|
template <typename T, std::size_t n_bits>
|
||||||
|
class endian_buffer< order::big, T, n_bits, align::no >
|
||||||
|
{
|
||||||
|
BOOST_STATIC_ASSERT( (n_bits/8)*8 == n_bits );
|
||||||
|
public:
|
||||||
|
typedef T value_type;
|
||||||
|
# ifndef BOOST_ENDIAN_NO_CTORS
|
||||||
|
endian_buffer() BOOST_ENDIAN_DEFAULT_CONSTRUCT
|
||||||
|
BOOST_ENDIAN_EXPLICIT_OPT endian_buffer(T val) BOOST_NOEXCEPT
|
||||||
|
{
|
||||||
|
# ifdef BOOST_ENDIAN_LOG
|
||||||
|
if ( endian_log )
|
||||||
|
std::cout << "big, unaligned, "
|
||||||
|
<< n_bits << "-bits, construct(" << val << ")\n";
|
||||||
|
# endif
|
||||||
|
detail::store_big_endian<T, n_bits/8>(m_value, val);
|
||||||
|
}
|
||||||
|
# endif
|
||||||
|
endian_buffer & operator=(T val) BOOST_NOEXCEPT
|
||||||
|
{
|
||||||
|
# ifdef BOOST_ENDIAN_LOG
|
||||||
|
if (endian_log)
|
||||||
|
std::cout << "big, unaligned, " << n_bits << "-bits, assign(" << val << ")\n";
|
||||||
|
# endif
|
||||||
|
detail::store_big_endian<T, n_bits/8>(m_value, val);
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
operator T() const BOOST_NOEXCEPT
|
||||||
|
{
|
||||||
|
# ifdef BOOST_ENDIAN_LOG
|
||||||
|
if ( endian_log )
|
||||||
|
std::cout << "big, unaligned, " << n_bits << "-bits, convert("
|
||||||
|
<< detail::load_big_endian<T, n_bits/8>(m_value) << ")\n";
|
||||||
|
# endif
|
||||||
|
return detail::load_big_endian<T, n_bits/8>(m_value);
|
||||||
|
}
|
||||||
|
const char* data() const BOOST_NOEXCEPT { return m_value; }
|
||||||
|
private:
|
||||||
|
char m_value[n_bits/8];
|
||||||
|
};
|
||||||
|
|
||||||
|
// unaligned float big endian_buffer specialization
|
||||||
|
template <>
|
||||||
|
class endian_buffer< order::big, float, 32, align::no >
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
typedef float value_type;
|
||||||
|
# ifndef BOOST_ENDIAN_NO_CTORS
|
||||||
|
endian_buffer() BOOST_ENDIAN_DEFAULT_CONSTRUCT
|
||||||
|
BOOST_ENDIAN_EXPLICIT_OPT endian_buffer(value_type val) BOOST_NOEXCEPT
|
||||||
|
{ detail::big_reverse_copy(val, m_value); }
|
||||||
|
# endif
|
||||||
|
endian_buffer & operator=(value_type val) BOOST_NOEXCEPT
|
||||||
|
{ detail::big_reverse_copy(val, m_value); return *this; }
|
||||||
|
operator value_type() const BOOST_NOEXCEPT
|
||||||
|
{
|
||||||
|
value_type tmp;
|
||||||
|
detail::big_reverse_copy(m_value, tmp);
|
||||||
|
return tmp;
|
||||||
|
}
|
||||||
|
const char* data() const BOOST_NOEXCEPT { return m_value; }
|
||||||
|
private:
|
||||||
|
char m_value[sizeof(value_type)];
|
||||||
|
};
|
||||||
|
|
||||||
|
// unaligned double big endian_buffer specialization
|
||||||
|
template <>
|
||||||
|
class endian_buffer< order::big, double, 64, align::no >
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
typedef double value_type;
|
||||||
|
# ifndef BOOST_ENDIAN_NO_CTORS
|
||||||
|
endian_buffer() BOOST_ENDIAN_DEFAULT_CONSTRUCT
|
||||||
|
BOOST_ENDIAN_EXPLICIT_OPT endian_buffer(value_type val) BOOST_NOEXCEPT
|
||||||
|
{ detail::big_reverse_copy(val, m_value); }
|
||||||
|
# endif
|
||||||
|
endian_buffer & operator=(value_type val) BOOST_NOEXCEPT
|
||||||
|
{ detail::big_reverse_copy(val, m_value); return *this; }
|
||||||
|
operator value_type() const BOOST_NOEXCEPT
|
||||||
|
{
|
||||||
|
value_type tmp;
|
||||||
|
detail::big_reverse_copy(m_value, tmp);
|
||||||
|
return tmp;
|
||||||
|
}
|
||||||
|
const char* data() const BOOST_NOEXCEPT { return m_value; }
|
||||||
|
private:
|
||||||
|
char m_value[sizeof(value_type)];
|
||||||
|
};
|
||||||
|
|
||||||
|
// unaligned float little endian_buffer specialization
|
||||||
|
template <>
|
||||||
|
class endian_buffer< order::little, float, 32, align::no >
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
typedef float value_type;
|
||||||
|
# ifndef BOOST_ENDIAN_NO_CTORS
|
||||||
|
endian_buffer() BOOST_ENDIAN_DEFAULT_CONSTRUCT
|
||||||
|
BOOST_ENDIAN_EXPLICIT_OPT endian_buffer(value_type val) BOOST_NOEXCEPT
|
||||||
|
{ detail::little_reverse_copy(val, m_value); }
|
||||||
|
# endif
|
||||||
|
endian_buffer & operator=(value_type val) BOOST_NOEXCEPT
|
||||||
|
{ detail::little_reverse_copy(val, m_value); return *this; }
|
||||||
|
operator value_type() const BOOST_NOEXCEPT
|
||||||
|
{
|
||||||
|
value_type tmp;
|
||||||
|
detail::little_reverse_copy(m_value, tmp);
|
||||||
|
return tmp;
|
||||||
|
}
|
||||||
|
const char* data() const BOOST_NOEXCEPT { return m_value; }
|
||||||
|
private:
|
||||||
|
char m_value[sizeof(value_type)];
|
||||||
|
};
|
||||||
|
|
||||||
|
// unaligned double little endian_buffer specialization
|
||||||
|
template <>
|
||||||
|
class endian_buffer< order::little, double, 64, align::no >
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
typedef double value_type;
|
||||||
|
# ifndef BOOST_ENDIAN_NO_CTORS
|
||||||
|
endian_buffer() BOOST_ENDIAN_DEFAULT_CONSTRUCT
|
||||||
|
BOOST_ENDIAN_EXPLICIT_OPT endian_buffer(value_type val) BOOST_NOEXCEPT
|
||||||
|
{ detail::little_reverse_copy(val, m_value); }
|
||||||
|
# endif
|
||||||
|
endian_buffer & operator=(value_type val) BOOST_NOEXCEPT
|
||||||
|
{ detail::little_reverse_copy(val, m_value); return *this; }
|
||||||
|
operator value_type() const BOOST_NOEXCEPT
|
||||||
|
{
|
||||||
|
value_type tmp;
|
||||||
|
detail::little_reverse_copy(m_value, tmp);
|
||||||
|
return tmp;
|
||||||
|
}
|
||||||
|
const char* data() const BOOST_NOEXCEPT { return m_value; }
|
||||||
|
private:
|
||||||
|
char m_value[sizeof(value_type)];
|
||||||
|
};
|
||||||
|
|
||||||
|
// unaligned little endian_buffer specialization
|
||||||
|
template <typename T, std::size_t n_bits>
|
||||||
|
class endian_buffer< order::little, T, n_bits, align::no >
|
||||||
|
{
|
||||||
|
BOOST_STATIC_ASSERT( (n_bits/8)*8 == n_bits );
|
||||||
|
public:
|
||||||
|
typedef T value_type;
|
||||||
|
# ifndef BOOST_ENDIAN_NO_CTORS
|
||||||
|
endian_buffer() BOOST_ENDIAN_DEFAULT_CONSTRUCT
|
||||||
|
BOOST_ENDIAN_EXPLICIT_OPT endian_buffer(T val) BOOST_NOEXCEPT
|
||||||
|
{
|
||||||
|
# ifdef BOOST_ENDIAN_LOG
|
||||||
|
if ( endian_log )
|
||||||
|
std::cout << "little, unaligned, " << n_bits << "-bits, construct("
|
||||||
|
<< val << ")\n";
|
||||||
|
# endif
|
||||||
|
detail::store_little_endian<T, n_bits/8>(m_value, val);
|
||||||
|
}
|
||||||
|
# endif
|
||||||
|
endian_buffer & operator=(T val) BOOST_NOEXCEPT
|
||||||
|
{ detail::store_little_endian<T, n_bits/8>(m_value, val); return *this; }
|
||||||
|
operator T() const BOOST_NOEXCEPT
|
||||||
|
{
|
||||||
|
# ifdef BOOST_ENDIAN_LOG
|
||||||
|
if ( endian_log )
|
||||||
|
std::cout << "little, unaligned, " << n_bits << "-bits, convert("
|
||||||
|
<< detail::load_little_endian<T, n_bits/8>(m_value) << ")\n";
|
||||||
|
# endif
|
||||||
|
return detail::load_little_endian<T, n_bits/8>(m_value);
|
||||||
|
}
|
||||||
|
const char* data() const BOOST_NOEXCEPT { return m_value; }
|
||||||
|
private:
|
||||||
|
char m_value[n_bits/8];
|
||||||
|
};
|
||||||
|
|
||||||
|
// align::yes specializations; only n_bits == 16/32/64 supported
|
||||||
|
|
||||||
|
// aligned big endian_buffer specialization
|
||||||
|
template <typename T, std::size_t n_bits>
|
||||||
|
class endian_buffer<order::big, T, n_bits, align::yes>
|
||||||
|
{
|
||||||
|
BOOST_STATIC_ASSERT( (n_bits/8)*8 == n_bits );
|
||||||
|
BOOST_STATIC_ASSERT( sizeof(T) == n_bits/8 );
|
||||||
|
public:
|
||||||
|
typedef T value_type;
|
||||||
|
# ifndef BOOST_ENDIAN_NO_CTORS
|
||||||
|
endian_buffer() BOOST_ENDIAN_DEFAULT_CONSTRUCT
|
||||||
|
BOOST_ENDIAN_EXPLICIT_OPT endian_buffer(T val) BOOST_NOEXCEPT
|
||||||
|
{
|
||||||
|
# ifdef BOOST_ENDIAN_LOG
|
||||||
|
if ( endian_log )
|
||||||
|
std::cout << "big, aligned, " << n_bits
|
||||||
|
<< "-bits, construct(" << val << ")\n";
|
||||||
|
# endif
|
||||||
|
m_value = ::boost::endian_buffer::big_endian_value(val);
|
||||||
|
}
|
||||||
|
|
||||||
|
# endif
|
||||||
|
endian_buffer& operator=(T val) BOOST_NOEXCEPT
|
||||||
|
{
|
||||||
|
m_value = ::boost::endian_buffer::big_endian_value(val);
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
operator T() const BOOST_NOEXCEPT
|
||||||
|
{
|
||||||
|
# ifdef BOOST_ENDIAN_LOG
|
||||||
|
if ( endian_log )
|
||||||
|
std::cout << "big, aligned, " << n_bits << "-bits, convert("
|
||||||
|
<< ::boost::endian_buffer::big_endian_value(m_value) << ")\n";
|
||||||
|
# endif
|
||||||
|
return ::boost::endian_buffer::big_endian_value(m_value);
|
||||||
|
}
|
||||||
|
const char* data() const BOOST_NOEXCEPT {return reinterpret_cast<const char*>(&m_value);}
|
||||||
|
private:
|
||||||
|
T m_value;
|
||||||
|
};
|
||||||
|
|
||||||
|
// aligned little endian_buffer specialization
|
||||||
|
template <typename T, std::size_t n_bits>
|
||||||
|
class endian_buffer<order::little, T, n_bits, align::yes>
|
||||||
|
{
|
||||||
|
BOOST_STATIC_ASSERT( (n_bits/8)*8 == n_bits );
|
||||||
|
BOOST_STATIC_ASSERT( sizeof(T) == n_bits/8 );
|
||||||
|
public:
|
||||||
|
typedef T value_type;
|
||||||
|
# ifndef BOOST_ENDIAN_NO_CTORS
|
||||||
|
endian_buffer() BOOST_ENDIAN_DEFAULT_CONSTRUCT
|
||||||
|
BOOST_ENDIAN_EXPLICIT_OPT endian_buffer(T val) BOOST_NOEXCEPT
|
||||||
|
{
|
||||||
|
# ifdef BOOST_ENDIAN_LOG
|
||||||
|
if ( endian_log )
|
||||||
|
std::cout << "little, aligned, " << n_bits
|
||||||
|
<< "-bits, construct(" << val << ")\n";
|
||||||
|
# endif
|
||||||
|
m_value = ::boost::endian_buffer::little_endian_value(val);
|
||||||
|
}
|
||||||
|
|
||||||
|
# endif
|
||||||
|
endian_buffer& operator=(T val) BOOST_NOEXCEPT
|
||||||
|
{
|
||||||
|
m_value = ::boost::endian_buffer::little_endian_value(val);
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
operator T() const BOOST_NOEXCEPT
|
||||||
|
{
|
||||||
|
# ifdef BOOST_ENDIAN_LOG
|
||||||
|
if ( endian_log )
|
||||||
|
std::cout << "little, aligned, " << n_bits << "-bits, convert("
|
||||||
|
<< ::boost::endian_buffer::little_endian_value(m_value) << ")\n";
|
||||||
|
# endif
|
||||||
|
return ::boost::endian_buffer::little_endian_value(m_value);
|
||||||
|
}
|
||||||
|
const char* data() const BOOST_NOEXCEPT {return reinterpret_cast<const char*>(&m_value);}
|
||||||
|
private:
|
||||||
|
T m_value;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace endian
|
||||||
|
} // namespace boost
|
||||||
|
|
||||||
|
#if defined(__BORLANDC__) || defined( __CODEGEARC__)
|
||||||
|
# pragma pack(pop)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if defined(_MSC_VER)
|
||||||
|
# pragma warning(pop)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif // BOOST_ENDIAN_BUFFERS_HPP
|
46
test/buffer_test.cpp
Normal file
46
test/buffer_test.cpp
Normal file
@@ -0,0 +1,46 @@
|
|||||||
|
// buffer_test.cpp -------------------------------------------------------------------//
|
||||||
|
|
||||||
|
// Copyright Beman Dawes 2014
|
||||||
|
|
||||||
|
// Distributed under the Boost Software License, Version 1.0.
|
||||||
|
// See http://www.boost.org/LICENSE_1_0.txt
|
||||||
|
|
||||||
|
// See library home page at http://www.boost.org/libs/endian
|
||||||
|
|
||||||
|
//--------------------------------------------------------------------------------------//
|
||||||
|
|
||||||
|
#define _SCL_SECURE_NO_WARNINGS
|
||||||
|
|
||||||
|
#include <boost/endian/detail/disable_warnings.hpp>
|
||||||
|
|
||||||
|
#define BOOST_ENDIAN_LOG
|
||||||
|
#include <boost/endian/buffers.hpp>
|
||||||
|
#include <boost/detail/lightweight_main.hpp>
|
||||||
|
#include <boost/detail/lightweight_test.hpp>
|
||||||
|
#include <iostream>
|
||||||
|
|
||||||
|
namespace bel = boost::endian;
|
||||||
|
using std::cout;
|
||||||
|
using std::endl;
|
||||||
|
|
||||||
|
|
||||||
|
int cpp_main(int, char *[])
|
||||||
|
{
|
||||||
|
cout << "byte swap intrinsics: " BOOST_ENDIAN_INTRINSIC_MSG << endl;
|
||||||
|
|
||||||
|
cout << " construct" << endl;
|
||||||
|
bel::big_buf32_t x(1122334455);
|
||||||
|
|
||||||
|
cout << " assign from built-in integer" << endl;
|
||||||
|
x = 1234567890;
|
||||||
|
|
||||||
|
cout << " operator==(buffer, built-in)" << endl;
|
||||||
|
bool b1(x == 1234567890);
|
||||||
|
|
||||||
|
// BOOST_TEST(x == 1234567890);
|
||||||
|
cout << " done" << endl;
|
||||||
|
|
||||||
|
return ::boost::report_errors();
|
||||||
|
}
|
||||||
|
|
||||||
|
#include <boost/endian/detail/disable_warnings_pop.hpp>
|
88
test/msvc/buffer_test/buffer_test.vcxproj
Normal file
88
test/msvc/buffer_test/buffer_test.vcxproj
Normal file
@@ -0,0 +1,88 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<Project DefaultTargets="Build" ToolsVersion="14.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
|
||||||
|
<ItemGroup Label="ProjectConfigurations">
|
||||||
|
<ProjectConfiguration Include="Debug|Win32">
|
||||||
|
<Configuration>Debug</Configuration>
|
||||||
|
<Platform>Win32</Platform>
|
||||||
|
</ProjectConfiguration>
|
||||||
|
<ProjectConfiguration Include="Release|Win32">
|
||||||
|
<Configuration>Release</Configuration>
|
||||||
|
<Platform>Win32</Platform>
|
||||||
|
</ProjectConfiguration>
|
||||||
|
</ItemGroup>
|
||||||
|
<PropertyGroup Label="Globals">
|
||||||
|
<ProjectGuid>{BFB68CF4-EB92-4E5C-9694-A939496C5CDE}</ProjectGuid>
|
||||||
|
<Keyword>Win32Proj</Keyword>
|
||||||
|
<RootNamespace>buffer_test</RootNamespace>
|
||||||
|
</PropertyGroup>
|
||||||
|
<Import Project="$(VCTargetsPath)\Microsoft.Cpp.Default.props" />
|
||||||
|
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'" Label="Configuration">
|
||||||
|
<ConfigurationType>Application</ConfigurationType>
|
||||||
|
<UseDebugLibraries>true</UseDebugLibraries>
|
||||||
|
<PlatformToolset>v140</PlatformToolset>
|
||||||
|
<CharacterSet>Unicode</CharacterSet>
|
||||||
|
</PropertyGroup>
|
||||||
|
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|Win32'" Label="Configuration">
|
||||||
|
<ConfigurationType>Application</ConfigurationType>
|
||||||
|
<UseDebugLibraries>false</UseDebugLibraries>
|
||||||
|
<PlatformToolset>v140</PlatformToolset>
|
||||||
|
<WholeProgramOptimization>true</WholeProgramOptimization>
|
||||||
|
<CharacterSet>Unicode</CharacterSet>
|
||||||
|
</PropertyGroup>
|
||||||
|
<Import Project="$(VCTargetsPath)\Microsoft.Cpp.props" />
|
||||||
|
<ImportGroup Label="ExtensionSettings">
|
||||||
|
</ImportGroup>
|
||||||
|
<ImportGroup Label="PropertySheets" Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">
|
||||||
|
<Import Project="$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props" Condition="exists('$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props')" Label="LocalAppDataPlatform" />
|
||||||
|
<Import Project="..\common.props" />
|
||||||
|
</ImportGroup>
|
||||||
|
<ImportGroup Label="PropertySheets" Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">
|
||||||
|
<Import Project="$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props" Condition="exists('$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props')" Label="LocalAppDataPlatform" />
|
||||||
|
<Import Project="..\common.props" />
|
||||||
|
</ImportGroup>
|
||||||
|
<PropertyGroup Label="UserMacros" />
|
||||||
|
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">
|
||||||
|
<LinkIncremental>true</LinkIncremental>
|
||||||
|
</PropertyGroup>
|
||||||
|
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">
|
||||||
|
<LinkIncremental>false</LinkIncremental>
|
||||||
|
</PropertyGroup>
|
||||||
|
<ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">
|
||||||
|
<ClCompile>
|
||||||
|
<PrecompiledHeader>
|
||||||
|
</PrecompiledHeader>
|
||||||
|
<WarningLevel>Level3</WarningLevel>
|
||||||
|
<Optimization>Disabled</Optimization>
|
||||||
|
<PreprocessorDefinitions>WIN32;_DEBUG;_CONSOLE;%(PreprocessorDefinitions)</PreprocessorDefinitions>
|
||||||
|
<SDLCheck>true</SDLCheck>
|
||||||
|
</ClCompile>
|
||||||
|
<Link>
|
||||||
|
<SubSystem>Console</SubSystem>
|
||||||
|
<GenerateDebugInformation>true</GenerateDebugInformation>
|
||||||
|
</Link>
|
||||||
|
</ItemDefinitionGroup>
|
||||||
|
<ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">
|
||||||
|
<ClCompile>
|
||||||
|
<WarningLevel>Level3</WarningLevel>
|
||||||
|
<PrecompiledHeader>
|
||||||
|
</PrecompiledHeader>
|
||||||
|
<Optimization>MaxSpeed</Optimization>
|
||||||
|
<FunctionLevelLinking>true</FunctionLevelLinking>
|
||||||
|
<IntrinsicFunctions>true</IntrinsicFunctions>
|
||||||
|
<PreprocessorDefinitions>WIN32;NDEBUG;_CONSOLE;%(PreprocessorDefinitions)</PreprocessorDefinitions>
|
||||||
|
<SDLCheck>true</SDLCheck>
|
||||||
|
</ClCompile>
|
||||||
|
<Link>
|
||||||
|
<SubSystem>Console</SubSystem>
|
||||||
|
<GenerateDebugInformation>true</GenerateDebugInformation>
|
||||||
|
<EnableCOMDATFolding>true</EnableCOMDATFolding>
|
||||||
|
<OptimizeReferences>true</OptimizeReferences>
|
||||||
|
</Link>
|
||||||
|
</ItemDefinitionGroup>
|
||||||
|
<ItemGroup>
|
||||||
|
<ClCompile Include="..\..\buffer_test.cpp" />
|
||||||
|
</ItemGroup>
|
||||||
|
<Import Project="$(VCTargetsPath)\Microsoft.Cpp.targets" />
|
||||||
|
<ImportGroup Label="ExtensionTargets">
|
||||||
|
</ImportGroup>
|
||||||
|
</Project>
|
@@ -1,6 +1,8 @@
|
|||||||
|
|
||||||
Microsoft Visual Studio Solution File, Format Version 12.00
|
Microsoft Visual Studio Solution File, Format Version 12.00
|
||||||
# Visual Studio Express 2012 for Windows Desktop
|
# Visual Studio 14
|
||||||
|
VisualStudioVersion = 14.0.22310.1
|
||||||
|
MinimumVisualStudioVersion = 10.0.40219.1
|
||||||
Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "endian_test", "endian_test\endian_test.vcxproj", "{74C201F3-8308-40BE-BC0F-24974DEAF405}"
|
Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "endian_test", "endian_test\endian_test.vcxproj", "{74C201F3-8308-40BE-BC0F-24974DEAF405}"
|
||||||
EndProject
|
EndProject
|
||||||
Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "endian_in_union_test", "endian_in_union_test\endian_in_union_test.vcxproj", "{3926C6DC-9D1E-4227-BEF5-81F5EC621A75}"
|
Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "endian_in_union_test", "endian_in_union_test\endian_in_union_test.vcxproj", "{3926C6DC-9D1E-4227-BEF5-81F5EC621A75}"
|
||||||
@@ -23,6 +25,8 @@ Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "speed_test", "speed_test\sp
|
|||||||
EndProject
|
EndProject
|
||||||
Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "loop_time_test", "loop_time_test\loop_time_test.vcxproj", "{541A2D06-B34E-4592-BE47-F87DF47E73D8}"
|
Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "loop_time_test", "loop_time_test\loop_time_test.vcxproj", "{541A2D06-B34E-4592-BE47-F87DF47E73D8}"
|
||||||
EndProject
|
EndProject
|
||||||
|
Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "buffer_test", "buffer_test\buffer_test.vcxproj", "{BFB68CF4-EB92-4E5C-9694-A939496C5CDE}"
|
||||||
|
EndProject
|
||||||
Global
|
Global
|
||||||
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
||||||
Debug|Win32 = Debug|Win32
|
Debug|Win32 = Debug|Win32
|
||||||
@@ -115,6 +119,12 @@ Global
|
|||||||
{541A2D06-B34E-4592-BE47-F87DF47E73D8}.Release|Win32.Build.0 = Release|Win32
|
{541A2D06-B34E-4592-BE47-F87DF47E73D8}.Release|Win32.Build.0 = Release|Win32
|
||||||
{541A2D06-B34E-4592-BE47-F87DF47E73D8}.Release|x64.ActiveCfg = Release|x64
|
{541A2D06-B34E-4592-BE47-F87DF47E73D8}.Release|x64.ActiveCfg = Release|x64
|
||||||
{541A2D06-B34E-4592-BE47-F87DF47E73D8}.Release|x64.Build.0 = Release|x64
|
{541A2D06-B34E-4592-BE47-F87DF47E73D8}.Release|x64.Build.0 = Release|x64
|
||||||
|
{BFB68CF4-EB92-4E5C-9694-A939496C5CDE}.Debug|Win32.ActiveCfg = Debug|Win32
|
||||||
|
{BFB68CF4-EB92-4E5C-9694-A939496C5CDE}.Debug|Win32.Build.0 = Debug|Win32
|
||||||
|
{BFB68CF4-EB92-4E5C-9694-A939496C5CDE}.Debug|x64.ActiveCfg = Debug|Win32
|
||||||
|
{BFB68CF4-EB92-4E5C-9694-A939496C5CDE}.Release|Win32.ActiveCfg = Release|Win32
|
||||||
|
{BFB68CF4-EB92-4E5C-9694-A939496C5CDE}.Release|Win32.Build.0 = Release|Win32
|
||||||
|
{BFB68CF4-EB92-4E5C-9694-A939496C5CDE}.Release|x64.ActiveCfg = Release|Win32
|
||||||
EndGlobalSection
|
EndGlobalSection
|
||||||
GlobalSection(SolutionProperties) = preSolution
|
GlobalSection(SolutionProperties) = preSolution
|
||||||
HideSolutionNode = FALSE
|
HideSolutionNode = FALSE
|
||||||
|
Reference in New Issue
Block a user