Boost logo

Boost-Commit :

Subject: [Boost-commit] svn:boost r71421 - in sandbox/endian: boost libs/integer/doc libs/integer/test
From: bdawes_at_[hidden]
Date: 2011-04-22 16:34:31


Author: bemandawes
Date: 2011-04-22 16:34:30 EDT (Fri, 22 Apr 2011)
New Revision: 71421
URL: http://svn.boost.org/trac/boost/changeset/71421

Log:
A fresh approach for binary_stream.hpp
Added:
   sandbox/endian/libs/integer/doc/binary_stream.html (contents, props changed)
Text files modified:
   sandbox/endian/boost/binary_stream.hpp | 157 ++++++++++-----------------------------
   sandbox/endian/libs/integer/test/binary_stream_test.cpp | 133 +++++++++++++++------------------
   2 files changed, 104 insertions(+), 186 deletions(-)

Modified: sandbox/endian/boost/binary_stream.hpp
==============================================================================
--- sandbox/endian/boost/binary_stream.hpp (original)
+++ sandbox/endian/boost/binary_stream.hpp 2011-04-22 16:34:30 EDT (Fri, 22 Apr 2011)
@@ -1,6 +1,6 @@
 // boost/binary_stream.hpp ----------------------------------------------------------//
 
-// Copyright Beman Dawes 2009
+// Copyright Beman Dawes 2009, 2011
 
 // Distributed under the Boost Software License, Version 1.0.
 // See http://www.boost.org/LICENSE_1_0.txt
@@ -20,7 +20,7 @@
 # include <cwchar> // for wcslen
 #endif
 
-// unformatted binary (as opposed to formatted character-set) input and output
+// unformatted binary (as opposed to formatted character) input and output
 
 // Caution: Use only on streams opened with filemode std::ios_base::binary. Thus
 // unformatted binary I/O should not be with the standard streams (cout, cin, etc.)
@@ -28,127 +28,54 @@
 // results, such as insertion of unwanted characters or premature end-of-file.
 // For example, on Windows 0x0D would become 0x0D, 0x0A.
 
-// Caution: When mixing formatted (i.e. operator << or >>) and unformatted (i.e.
-// operator <= or >=) be aware that << and >> take precedence over <= and >=. Use
-// parentheses to force correct order of evaluation. For example:
-//
-// my_stream << foo <= bar; // no parentheses needed
-// (my_stream <= foo) << bar; // parentheses required
-
-// This implementation uses reinterpret_cast<>() when needed to convert one pointer
-// type to another. See 5.2.10 [expr.reinterpret.cast], Reinterpret cast, para 7.
-
 namespace boost
 {
 
- // built-in types ------------------------------------------------------------------//
-
- // omission of bool and void* is deliberate; any semantics would be questionable
-
- inline std::ostream& operator<=(std::ostream& os, short v)
- { return os.write( reinterpret_cast<const char*>(&v), sizeof(v) ); }
- inline std::istream& operator>=(std::istream& is, short& v)
- { return is.read( reinterpret_cast<char*>(&v), sizeof(v) ); }
-
- inline std::ostream& operator<=(std::ostream& os, unsigned short v)
- { return os.write( reinterpret_cast<const char*>(&v), sizeof(v) ); }
- inline std::istream& operator>=(std::istream& is, unsigned short& v)
- { return is.read( reinterpret_cast<char*>(&v), sizeof(v) ); }
-
- inline std::ostream& operator<=(std::ostream& os, int v)
- { return os.write( reinterpret_cast<const char*>(&v), sizeof(v) ); }
- inline std::istream& operator>=(std::istream& is, int& v)
- { return is.read( reinterpret_cast<char*>(&v), sizeof(v) ); }
-
- inline std::ostream& operator<=(std::ostream& os, unsigned int v)
- { return os.write( reinterpret_cast<const char*>(&v), sizeof(v) ); }
- inline std::istream& operator>=(std::istream& is, unsigned int& v)
- { return is.read( reinterpret_cast<char*>(&v), sizeof(v) ); }
-
- inline std::ostream& operator<=(std::ostream& os, long v)
- { return os.write( reinterpret_cast<const char*>(&v), sizeof(v) ); }
- inline std::istream& operator>=(std::istream& is, long& v)
- { return is.read( reinterpret_cast<char*>(&v), sizeof(v) ); }
-
- inline std::ostream& operator<=(std::ostream& os, unsigned long v)
- { return os.write( reinterpret_cast<const char*>(&v), sizeof(v) ); }
- inline std::istream& operator>=(std::istream& is, unsigned long& v)
- { return is.read( reinterpret_cast<char*>(&v), sizeof(v) ); }
-
- inline std::ostream& operator<=(std::ostream& os, long long v)
- { return os.write( reinterpret_cast<const char*>(&v), sizeof(v) ); }
- inline std::istream& operator>=(std::istream& is, long long& v)
- { return is.read( reinterpret_cast<char*>(&v), sizeof(v) ); }
-
- inline std::ostream& operator<=(std::ostream& os, unsigned long long v)
- { return os.write( reinterpret_cast<const char*>(&v), sizeof(v) ); }
- inline std::istream& operator>=(std::istream& is, unsigned long long& v)
- { return is.read( reinterpret_cast<char*>(&v), sizeof(v) ); }
-
- inline std::ostream& operator<=(std::ostream& os, float v)
- { return os.write( reinterpret_cast<const char*>(&v), sizeof(v) ); }
- inline std::istream& operator>=(std::istream& is, float& v)
- { return is.read( reinterpret_cast<char*>(&v), sizeof(v) ); }
-
- inline std::ostream& operator<=(std::ostream& os, double v)
- { return os.write( reinterpret_cast<const char*>(&v), sizeof(v) ); }
- inline std::istream& operator>=(std::istream& is, double& v)
- { return is.read( reinterpret_cast<char*>(&v), sizeof(v) ); }
-
- inline std::ostream& operator<=(std::ostream& os, long double v)
- { return os.write( reinterpret_cast<const char*>(&v), sizeof(v) ); }
- inline std::istream& operator>=(std::istream& is, long double& v)
- { return is.read( reinterpret_cast<char*>(&v), sizeof(v) ); }
-
- inline std::ostream& operator<=(std::ostream& os, char c)
- { return os.put( c ); }
- inline std::istream& operator>=(std::istream& is, char& c)
- { return is.get( c ); }
-
- inline std::ostream& operator<=(std::ostream& os, signed char c)
- { return os.put( c ); }
- inline std::istream& operator>=(std::istream& is, signed char& c)
- { return is.get( reinterpret_cast<char&>(c) ); }
-
- inline std::ostream& operator<=(std::ostream& os, unsigned char c)
- { return os.put( c ); }
- inline std::istream& operator>=(std::istream& is, unsigned char& c)
- { return is.get( reinterpret_cast<char&>(c) ); }
-
- inline std::ostream& operator<=(std::ostream& os, wchar_t v)
- { return os.write( reinterpret_cast<const char*>(&v), sizeof(v) ); }
- inline std::istream& operator>=(std::istream& is, wchar_t& v)
- { return is.read( reinterpret_cast<char*>(&v), sizeof(v) ); }
-
- // strings -------------------------------------------------------------------------//
+namespace detail
+{
+ template <class T>
+ struct const_binary_data
+ {
+ const char* ptr;
+ explicit const_binary_data(const T& x) : ptr(reinterpret_cast<const char*>(&x)) {}
+ };
+ template <class T>
+ struct binary_data
+ {
+ char* ptr;
+ explicit binary_data(T& x) : ptr(reinterpret_cast<char*>(&x)) {}
+ };
+}
 
- inline std::ostream& operator<=(std::ostream& os, const char* p)
- { return os.write( p, std::strlen(p)+1 ); }
+template <class T>
+inline detail::const_binary_data<T> bin(const T& x)
+{
+ return detail::const_binary_data<T>(x);
+}
 
- inline std::ostream& operator<=(std::ostream& os, const signed char* p)
- { return os.write( reinterpret_cast<const char*>(p), std::strlen(reinterpret_cast<const char*>(p))+1 ); }
+template <class T>
+inline detail::binary_data<T> bin(T& x)
+{
+ return detail::binary_data<T>(x);
+}
 
- inline std::ostream& operator<=(std::ostream& os, const unsigned char* p)
- { return os.write( reinterpret_cast<const char*>(p), std::strlen(reinterpret_cast<const char*>(p))+1 ); }
+template <class T>
+inline std::ostream& operator<<(std::ostream& os, detail::const_binary_data<T> x)
+{
+ return os.write(x.ptr, sizeof(T));
+}
 
-#ifndef BOOST_NO_CWCHAR
- inline std::ostream& operator<=(std::ostream& os, const wchar_t* p)
- { return os.write( reinterpret_cast<const char*>(p), (std::wcslen(p)+1)*sizeof(wchar_t) ); }
-#endif
+template <class T>
+inline std::ostream& operator<<(std::ostream& os, detail::binary_data<T> x)
+{
+ return os.write(x.ptr, sizeof(T));
+}
 
- // Caution: note the asymmetry between output and input; a string with embedded
- // nulls will be output with the embedded nulls, but input will stop at the first null.
- // So it probably isn't a good idea to use these functions for strings with nulls.
- inline std::ostream& operator<=(std::ostream& os, const std::string& s)
- { return os.write( s.c_str(), s.size()+1 ); }
- inline std::istream& operator>=(std::istream& is, std::string& s)
- { return getline(is, s, '\0'); }
-
-#ifndef BOOST_NO_STD_WSTRING
- inline std::ostream& operator<=(std::ostream& os, const std::wstring& s)
- { return os.write( reinterpret_cast<const char*>(s.c_str()), (s.size()+1)*sizeof(wchar_t) ); }
- // TODO: provide input function
-#endif
+template <class T>
+inline std::istream& operator>>(std::istream& is, detail::binary_data<T> x)
+{
+ return is.read(x.ptr, sizeof(T));
+}
 
 } // namespace boost
 

Added: sandbox/endian/libs/integer/doc/binary_stream.html
==============================================================================
--- (empty file)
+++ sandbox/endian/libs/integer/doc/binary_stream.html 2011-04-22 16:34:30 EDT (Fri, 22 Apr 2011)
@@ -0,0 +1,89 @@
+<html>
+
+<head>
+<meta http-equiv="Content-Language" content="en-us">
+<meta name="GENERATOR" content="Microsoft FrontPage 5.0">
+<meta name="ProgId" content="FrontPage.Editor.Document">
+<meta http-equiv="Content-Type" content="text/html; charset=windows-1252">
+<title>Binary Stream I/O</title>
+<link rel="stylesheet" type="text/css" href="../../../doc/html/minimal.css">
+</head>
+
+<body>
+
+<h1>Proposal for Enhanced Binary Stream I/O</h1>
+<h2>Introduction</h2>
+<p>The C++ standard library's stream I/O facilities are type-safe and very
+convenient for performing formatted (i.e. human readable) I/O, but offer only
+rudimentary and not very type-safe operations for performing binary I/O.&nbsp;
+Although formatted I/O is often preferable, some applications need the speed and
+storage efficiency of binary I/O or need to interoperate with third-party
+applications that require binary file or network data formats.</p>
+<p>Standard library streams can be opened with filemode <code>
+std::ios_base::binary</code>, so binary I/O is possible. But the only
+unformatted I/O functions available are <code>get()</code>, <code>put()</code>,
+<code>read()</code>, and <code>write()</code>. These operate only on <code>char</code>
+or array of <code>char</code> (with length explicitly specified), so require the
+user to write casts, are hard to use, and are error prone.</p>
+<p>There have been many requests on Boost and various C++ newsgroups for
+unformatted binary I/O. For example, in 2003 Neal Becker wrote:</p>
+<blockquote>
+<p>I wonder if anyone has code for implementing unformatted I/O?&nbsp; What I
+have in mind is for the simple case where the application that reads data knows
+the data types, so this is not as complicated as the general marshalling
+situation.</p>
+</blockquote>
+<p>This proposal provides a simple solution that works will standard library
+input and output streams. The one caveat is that they should be open with
+filemode <code>std::ios_base::binary</code>.</p>
+<h2>Synopsis</h2>
+<div dir="ltr">
+ <pre>namespace boost
+{
+ template &lt;class T&gt;
+ <i>unspecified-type-1&lt;T&gt;</i> bin(const T&amp; x);
+
+ template &lt;class T&gt;
+ <i>unspecified-type-2&lt;T&gt;</i> bin(T&amp; x);
+
+ template &lt;class T&gt;
+ std::ostream&amp; operator&lt;&lt;(std::ostream&amp; os, <i>unspecified-type-1&lt;T&gt;</i> x);
+
+ template &lt;class T&gt;
+ std::ostream&amp; operator&lt;&lt;(std::ostream&amp; os, <i>unspecified-type-2&lt;T&gt;</i> x);
+
+ template &lt;class T&gt;
+ std::istream&amp; operator&gt;&gt;(std::istream&amp; is, <i>unspecified-type-2&lt;T&gt;</i> x);
+}</pre>
+</div>
+<p><i><code>unspecified-type-1</code></i> and <i><code>unspecified-type-2</code></i>
+are implementation supplied types.</p>
+<h2>Example</h2>
+<blockquote>
+ <pre>int main()
+{
+ int i = 0x41424344;
+ std::cout &lt;&lt; std::hex &lt;&lt; i &lt;&lt; &quot; &quot; &lt;&lt; bin(i) &lt;&lt; '\n';
+ return 0;
+}</pre>
+</blockquote>
+<p>On a little-endian machine, the output is:</p>
+<blockquote>
+ <pre>41424344 DCBA</pre>
+</blockquote>
+<p>&nbsp;</p>
+<hr>
+<p>Last revised:
+<!--webbot bot="Timestamp" s-type="EDITED" s-format="%d %B, %Y" startspan -->22 April, 2011<!--webbot bot="Timestamp" endspan i-checksum="29826" --></p>
+<p>© Copyright Beman Dawes, 2009, 2011</p>
+<p>Distributed under the Boost Software License, Version 1.0. See
+www.boost.org/ LICENSE_1_0.txt</p>
+
+<p>&nbsp;</p>
+<p>&nbsp;</p>
+<p>&nbsp;</p>
+<p>&nbsp;</p>
+
+</body>
+
+</html>
\ No newline at end of file

Modified: sandbox/endian/libs/integer/test/binary_stream_test.cpp
==============================================================================
--- sandbox/endian/libs/integer/test/binary_stream_test.cpp (original)
+++ sandbox/endian/libs/integer/test/binary_stream_test.cpp 2011-04-22 16:34:30 EDT (Fri, 22 Apr 2011)
@@ -9,113 +9,104 @@
 #include <string>
 #include <iostream>
 #include <sstream>
+#include <boost/detail/lightweight_test.hpp>
 
 using namespace boost;
 using namespace std;
 
-namespace
-{
- int errors = 0;
-
- void verify( bool x, const char * file, int line )
- {
- if ( x ) return;
- ++errors;
- cout << file << "(" << line << ") : error: predicate is false\n" << endl;
- }
-
-}
-
-#define BOOST_VERIFY(predicate) verify( predicate, __FILE__, __LINE__ )
-
-
 int main()
 {
   std::stringstream ss( std::ios_base::in | std::ios_base::out | std::ios_base::binary );
 
   short short_1(0x0102), short_2;
- ss <= short_1;
- ss >= short_2;
- BOOST_VERIFY( short_1 == short_2 );
+ ss.clear();
+ ss << bin(short_1);
+ ss >> bin(short_2);
+ BOOST_TEST( short_1 == short_2 );
 
   unsigned short ushort_1(0x0102), ushort_2;
- ss <= ushort_1;
- ss >= ushort_2;
- BOOST_VERIFY( ushort_1 == ushort_2 );
+ ss.clear();
+ ss << bin(ushort_1);
+ ss >> bin(ushort_2);
+ BOOST_TEST( ushort_1 == ushort_2 );
 
   int int_1(0x01020304), int_2;
- ss <= int_1;
- ss >= int_2;
- BOOST_VERIFY( int_1 == int_2 );
+ ss.clear();
+ ss << bin(int_1);
+ ss >> bin(int_2);
+ BOOST_TEST( int_1 == int_2 );
 
   unsigned int uint_1(0x01020304), uint_2;
- ss <= uint_1;
- ss >= uint_2;
- BOOST_VERIFY( uint_1 == uint_2 );
+ ss.clear();
+ ss << bin(uint_1);
+ ss >> bin(uint_2);
+ BOOST_TEST( uint_1 == uint_2 );
 
   long long_1(0x01020304L), long_2;
- ss <= long_1;
- ss >= long_2;
- BOOST_VERIFY( long_1 == long_2 );
+ ss.clear();
+ ss << bin(long_1);
+ ss >> bin(long_2);
+ BOOST_TEST( long_1 == long_2 );
 
   unsigned long ulong_1(0x01020304UL), ulong_2;
- ss <= ulong_1;
- ss >= ulong_2;
- BOOST_VERIFY( ulong_1 == ulong_2 );
+ ss.clear();
+ ss << bin(ulong_1);
+ ss >> bin(ulong_2);
+ BOOST_TEST( ulong_1 == ulong_2 );
 
   long long long_long_1(0x0102030405060708LL), long_long_2;
- ss <= long_long_1;
- ss >= long_long_2;
- BOOST_VERIFY( long_long_1 == long_long_2 );
+ ss.clear();
+ ss << bin(long_long_1);
+ ss >> bin(long_long_2);
+ BOOST_TEST( long_long_1 == long_long_2 );
 
   unsigned long long ulong_long_1(0x0102030405060708ULL), ulong_long_2;
- ss <= ulong_long_1;
- ss >= ulong_long_2;
- BOOST_VERIFY( ulong_long_1 == ulong_long_2 );
+ ss.clear();
+ ss << bin(ulong_long_1);
+ ss >> bin(ulong_long_2);
+ BOOST_TEST( ulong_long_1 == ulong_long_2 );
 
   float float_1(1.2F), float_2;
- ss <= float_1;
- ss >= float_2;
- BOOST_VERIFY( float_1 == float_2 );
+ ss.clear();
+ ss << bin(float_1);
+ ss >> bin(float_2);
+ BOOST_TEST( float_1 == float_2 );
 
   double double_1(1.2), double_2;
- ss <= double_1;
- ss >= double_2;
- BOOST_VERIFY( double_1 == double_2 );
+ ss.clear();
+ ss << bin(double_1);
+ ss >> bin(double_2);
+ BOOST_TEST( double_1 == double_2 );
 
   long double long_double_1(1.2), long_double_2;
- ss <= long_double_1;
- ss >= long_double_2;
- BOOST_VERIFY( long_double_1 == long_double_2 );
+ ss.clear();
+ ss << bin(long_double_1);
+ ss >> bin(long_double_2);
+ BOOST_TEST( long_double_1 == long_double_2 );
 
   char char_1(0x01), char_2;
- ss <= char_1;
- ss >= char_2;
- BOOST_VERIFY( char_1 == char_2 );
+ ss.clear();
+ ss << bin(char_1);
+ ss >> bin(char_2);
+ BOOST_TEST( char_1 == char_2 );
 
   signed char schar_1(0x01), schar_2;
- ss <= schar_1;
- ss >= schar_2;
- BOOST_VERIFY( schar_1 == schar_2 );
+ ss.clear();
+ ss << bin(schar_1);
+ ss >> bin(schar_2);
+ BOOST_TEST( schar_1 == schar_2 );
 
   unsigned char uchar_1(0x01), uchar_2;
- ss <= uchar_1;
- ss >= uchar_2;
- BOOST_VERIFY( uchar_1 == uchar_2 );
+ ss.clear();
+ ss << bin(uchar_1);
+ ss >> bin(uchar_2);
+ BOOST_TEST( uchar_1 == uchar_2 );
 
   wchar_t wchar_t_1(L'1'), wchar_t_2;
- ss <= wchar_t_1;
- ss >= wchar_t_2;
- BOOST_VERIFY( wchar_t_1 == wchar_t_2 );
-
- string string_1("foobar"), string_2;
- ss <= string_1.c_str();
- ss >= string_2;
- BOOST_VERIFY( string_1 == string_2 );
- ss <= string_1;
- ss >= string_2;
- BOOST_VERIFY( string_1 == string_2 );
+ ss.clear();
+ ss << bin(wchar_t_1);
+ ss >> bin(wchar_t_2);
+ BOOST_TEST( wchar_t_1 == wchar_t_2 );
 
- cout << errors << " error(s) detected\n";
- return errors;
+ return ::boost::report_errors();
 }


Boost-Commit list run by bdawes at acm.org, david.abrahams at rcn.com, gregod at cs.rpi.edu, cpdaniel at pacbell.net, john at johnmaddock.co.uk