$include_dir="/home/hyper-archives/boost-commit/include"; include("$include_dir/msg-header.inc") ?>
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.  
+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?  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 <class T>
+  <i>unspecified-type-1<T></i> bin(const T& x);
+
+  template <class T>
+  <i>unspecified-type-2<T></i> bin(T& x);
+
+  template <class T>
+  std::ostream& operator<<(std::ostream& os, <i>unspecified-type-1<T></i> x);
+
+  template <class T>
+  std::ostream& operator<<(std::ostream& os, <i>unspecified-type-2<T></i> x);
+
+  template <class T>
+  std::istream& operator>>(std::istream& is, <i>unspecified-type-2<T></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 << std::hex << i << " " << bin(i) << '\n';
+  return 0;
+}</pre>
+</blockquote>
+<p>On a little-endian machine, the output is:</p>
+<blockquote>
+  <pre>41424344 DCBA</pre>
+</blockquote>
+<p> </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> </p>
+<p> </p>
+<p> </p>
+<p> </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();
 }