os/ossrv/ossrv_pub/boost_apis/boost/iostreams/copy.hpp
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
sl@0
     1
// (C) Copyright Jonathan Turkanis 2003.
sl@0
     2
// Distributed under the Boost Software License, Version 1.0. (See accompanying
sl@0
     3
// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt.)
sl@0
     4
sl@0
     5
// See http://www.boost.org/libs/iostreams for documentation.
sl@0
     6
sl@0
     7
// Contains: The function template copy, which reads data from a Source 
sl@0
     8
// and writes it to a Sink until the end of the sequence is reached, returning 
sl@0
     9
// the number of characters transfered.
sl@0
    10
sl@0
    11
// The implementation is complicated by the need to handle smart adapters
sl@0
    12
// and direct devices.
sl@0
    13
sl@0
    14
#ifndef BOOST_IOSTREAMS_COPY_HPP_INCLUDED
sl@0
    15
#define BOOST_IOSTREAMS_COPY_HPP_INCLUDED
sl@0
    16
sl@0
    17
#if defined(_MSC_VER) && (_MSC_VER >= 1020)
sl@0
    18
# pragma once
sl@0
    19
#endif              
sl@0
    20
sl@0
    21
#include <algorithm>                        // copy.
sl@0
    22
#include <utility>                          // pair.
sl@0
    23
#include <boost/detail/workaround.hpp>
sl@0
    24
#include <boost/iostreams/chain.hpp>
sl@0
    25
#include <boost/iostreams/constants.hpp>
sl@0
    26
#include <boost/iostreams/detail/adapter/non_blocking_adapter.hpp>        
sl@0
    27
#include <boost/iostreams/detail/buffer.hpp>       
sl@0
    28
#include <boost/iostreams/detail/closer.hpp>    
sl@0
    29
#include <boost/iostreams/detail/enable_if_stream.hpp>  
sl@0
    30
#include <boost/iostreams/detail/ios.hpp>   // failure, streamsize.                   
sl@0
    31
#include <boost/iostreams/detail/resolve.hpp>                   
sl@0
    32
#include <boost/iostreams/detail/wrap_unwrap.hpp>
sl@0
    33
#include <boost/iostreams/operations.hpp>  // read, write, close.
sl@0
    34
#include <boost/iostreams/pipeline.hpp>
sl@0
    35
#include <boost/static_assert.hpp>  
sl@0
    36
#include <boost/type_traits/is_same.hpp> 
sl@0
    37
sl@0
    38
namespace boost { namespace iostreams {
sl@0
    39
sl@0
    40
namespace detail {
sl@0
    41
sl@0
    42
template<typename Source, typename Sink>
sl@0
    43
std::streamsize copy_impl( Source& src, Sink& snk, 
sl@0
    44
                           std::streamsize /* buffer_size */,
sl@0
    45
                           mpl::true_, mpl::true_ )
sl@0
    46
{   // Copy from a direct Source to a direct Sink.
sl@0
    47
    using namespace std;
sl@0
    48
    typedef typename char_type_of<Source>::type  char_type;
sl@0
    49
    typedef pair<char_type*, char_type*>         pair_type;
sl@0
    50
    pair_type p1 = iostreams::input_sequence(src);
sl@0
    51
    pair_type p2 = iostreams::output_sequence(snk);
sl@0
    52
    if (p1.second - p1.first < p2.second - p2.first) {
sl@0
    53
        std::copy(p1.first, p1.second, p2.first);
sl@0
    54
        return static_cast<streamsize>(p1.second - p1.first);
sl@0
    55
    } else {
sl@0
    56
        throw BOOST_IOSTREAMS_FAILURE("destination too small");
sl@0
    57
    }
sl@0
    58
}
sl@0
    59
sl@0
    60
template<typename Source, typename Sink>
sl@0
    61
std::streamsize copy_impl( Source& src, Sink& snk, 
sl@0
    62
                           std::streamsize /* buffer_size */,
sl@0
    63
                           mpl::true_, mpl::false_ )
sl@0
    64
{   // Copy from a direct Source to an indirect Sink.
sl@0
    65
    using namespace std;
sl@0
    66
    typedef typename char_type_of<Source>::type  char_type;
sl@0
    67
    typedef pair<char_type*, char_type*>         pair_type;
sl@0
    68
    pair_type p = iostreams::input_sequence(src);
sl@0
    69
    std::streamsize size, total;
sl@0
    70
    for ( total = 0, size = static_cast<streamsize>(p.second - p.first);
sl@0
    71
          total < size; )
sl@0
    72
    {
sl@0
    73
        std::streamsize amt = 
sl@0
    74
            iostreams::write(snk, p.first + total, size - total); 
sl@0
    75
        total += amt;
sl@0
    76
    }
sl@0
    77
    return size;
sl@0
    78
}
sl@0
    79
sl@0
    80
template<typename Source, typename Sink>
sl@0
    81
std::streamsize copy_impl( Source& src, Sink& snk, 
sl@0
    82
                           std::streamsize buffer_size,
sl@0
    83
                           mpl::false_, mpl::true_ )
sl@0
    84
{   // Copy from an indirect Source to a direct Sink.
sl@0
    85
    using namespace std;
sl@0
    86
    typedef typename char_type_of<Source>::type  char_type;
sl@0
    87
    typedef pair<char_type*, char_type*>         pair_type;
sl@0
    88
    detail::basic_buffer<char_type>  buf(buffer_size);
sl@0
    89
    pair_type                        p = snk.output_sequence();
sl@0
    90
    streamsize                       total = 0;
sl@0
    91
    bool                             done  = false;
sl@0
    92
    while (!done) {
sl@0
    93
        streamsize amt;
sl@0
    94
        done = (amt = iostreams::read(src, buf.data(), buffer_size)) == -1;
sl@0
    95
        std::copy(buf.data(), buf.data() + amt, p.first + total);
sl@0
    96
        if (amt != -1)
sl@0
    97
            total += amt;
sl@0
    98
    }
sl@0
    99
    return total;
sl@0
   100
}
sl@0
   101
sl@0
   102
template<typename Source, typename Sink>
sl@0
   103
std::streamsize copy_impl( Source& src, Sink& snk, 
sl@0
   104
                           std::streamsize buffer_size,
sl@0
   105
                           mpl::false_, mpl::false_ )
sl@0
   106
{   // Copy from an indirect Source to a indirect Sink. This algorithm
sl@0
   107
    // can be improved by eliminating the non_blocking_adapter.
sl@0
   108
    typedef typename char_type_of<Source>::type char_type;
sl@0
   109
    detail::basic_buffer<char_type>  buf(buffer_size);
sl@0
   110
    non_blocking_adapter<Sink>       nb(snk);
sl@0
   111
    std::streamsize                  total = 0;
sl@0
   112
    bool                             done = false;
sl@0
   113
    while (!done) {
sl@0
   114
        std::streamsize amt;
sl@0
   115
        done = (amt = iostreams::read(src, buf.data(), buffer_size)) == -1;
sl@0
   116
        if (amt != -1) {
sl@0
   117
            iostreams::write(nb, buf.data(), amt);
sl@0
   118
            total += amt;
sl@0
   119
        }
sl@0
   120
    }
sl@0
   121
    return total;
sl@0
   122
}
sl@0
   123
sl@0
   124
template<typename Source, typename Sink>
sl@0
   125
std::streamsize copy_impl(Source src, Sink snk, std::streamsize buffer_size)
sl@0
   126
{
sl@0
   127
    using namespace std;
sl@0
   128
    typedef typename char_type_of<Source>::type  src_char;
sl@0
   129
    typedef typename char_type_of<Sink>::type    snk_char;
sl@0
   130
    BOOST_STATIC_ASSERT((is_same<src_char, snk_char>::value));
sl@0
   131
    bool                     nothrow = false;
sl@0
   132
    external_closer<Source>  close_source(src, BOOST_IOS::in, nothrow);
sl@0
   133
    external_closer<Sink>    close_sink(snk, BOOST_IOS::out, nothrow);
sl@0
   134
    streamsize result =
sl@0
   135
        copy_impl( src, snk, buffer_size, 
sl@0
   136
                   is_direct<Source>(), is_direct<Sink>() );
sl@0
   137
    return result; 
sl@0
   138
}
sl@0
   139
sl@0
   140
} // End namespace detail.
sl@0
   141
                    
sl@0
   142
//------------------Definition of copy----------------------------------------//
sl@0
   143
sl@0
   144
template<typename Source, typename Sink>
sl@0
   145
std::streamsize
sl@0
   146
copy( const Source& src, const Sink& snk,
sl@0
   147
      std::streamsize buffer_size = default_device_buffer_size
sl@0
   148
      BOOST_IOSTREAMS_DISABLE_IF_STREAM(Source)
sl@0
   149
      BOOST_IOSTREAMS_DISABLE_IF_STREAM(Sink) )
sl@0
   150
{ 
sl@0
   151
    typedef typename char_type_of<Source>::type char_type;
sl@0
   152
    return detail::copy_impl( detail::resolve<input, char_type>(src), 
sl@0
   153
                              detail::resolve<output, char_type>(snk), 
sl@0
   154
                              buffer_size ); 
sl@0
   155
}
sl@0
   156
sl@0
   157
#if !BOOST_WORKAROUND(BOOST_MSVC, <= 1300) //---------------------------------//
sl@0
   158
sl@0
   159
template<typename Source, typename Sink>
sl@0
   160
std::streamsize
sl@0
   161
copy( Source& src, const Sink& snk,
sl@0
   162
      std::streamsize buffer_size = default_device_buffer_size
sl@0
   163
      BOOST_IOSTREAMS_ENABLE_IF_STREAM(Source)
sl@0
   164
      BOOST_IOSTREAMS_DISABLE_IF_STREAM(Sink) ) 
sl@0
   165
{ 
sl@0
   166
    typedef typename char_type_of<Source>::type char_type;
sl@0
   167
    return detail::copy_impl( detail::wrap(src), 
sl@0
   168
                              detail::resolve<output, char_type>(snk), 
sl@0
   169
                              buffer_size );
sl@0
   170
}
sl@0
   171
sl@0
   172
template<typename Source, typename Sink>
sl@0
   173
std::streamsize
sl@0
   174
copy( const Source& src, Sink& snk,
sl@0
   175
      std::streamsize buffer_size = default_device_buffer_size
sl@0
   176
      BOOST_IOSTREAMS_DISABLE_IF_STREAM(Source)
sl@0
   177
      BOOST_IOSTREAMS_ENABLE_IF_STREAM(Sink) ) 
sl@0
   178
{ 
sl@0
   179
    typedef typename char_type_of<Source>::type char_type;
sl@0
   180
    return detail::copy_impl( detail::resolve<input, char_type>(src), 
sl@0
   181
                              detail::wrap(snk), buffer_size);
sl@0
   182
}
sl@0
   183
sl@0
   184
template<typename Source, typename Sink>
sl@0
   185
std::streamsize
sl@0
   186
copy( Source& src, Sink& snk,
sl@0
   187
      std::streamsize buffer_size = default_device_buffer_size
sl@0
   188
      BOOST_IOSTREAMS_ENABLE_IF_STREAM(Source)
sl@0
   189
      BOOST_IOSTREAMS_ENABLE_IF_STREAM(Sink) ) 
sl@0
   190
{ 
sl@0
   191
    return detail::copy_impl(detail::wrap(src), detail::wrap(snk), buffer_size);
sl@0
   192
}
sl@0
   193
sl@0
   194
#endif // #if !BOOST_WORKAROUND(BOOST_MSVC, <= 1300) //-----------------------//
sl@0
   195
sl@0
   196
} } // End namespaces iostreams, boost.
sl@0
   197
sl@0
   198
#endif // #ifndef BOOST_IOSTREAMS_COPY_HPP_INCLUDED