254 lines
7.1 KiB
C++
254 lines
7.1 KiB
C++
// (C) Copyright 2008 CodeRage, LLC (turkanis at coderage dot com)
|
|
// (C) Copyright 2003-2007 Jonathan Turkanis
|
|
// Distributed under the Boost Software License, Version 1.0. (See accompanying
|
|
// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt.)
|
|
|
|
// See http://www.boost.org/libs/iostreams for documentation.
|
|
|
|
#ifndef BOOST_IOSTREAMS_CLOSE_HPP_INCLUDED
|
|
#define BOOST_IOSTREAMS_CLOSE_HPP_INCLUDED
|
|
|
|
#if defined(_MSC_VER)
|
|
# pragma once
|
|
#endif
|
|
|
|
#include <boost/config.hpp> // DEDUCED_TYPENAME, MSVC.
|
|
#include <boost/detail/workaround.hpp>
|
|
#include <boost/iostreams/categories.hpp>
|
|
#include <boost/iostreams/flush.hpp>
|
|
#include <boost/iostreams/detail/adapter/non_blocking_adapter.hpp>
|
|
#include <boost/iostreams/detail/ios.hpp> // BOOST_IOS
|
|
#include <boost/iostreams/detail/select.hpp>
|
|
#include <boost/iostreams/detail/wrap_unwrap.hpp>
|
|
#include <boost/iostreams/operations_fwd.hpp>
|
|
#include <boost/iostreams/traits.hpp>
|
|
#include <boost/mpl/identity.hpp>
|
|
#include <boost/mpl/if.hpp>
|
|
#include <boost/type_traits/is_convertible.hpp>
|
|
#include <boost/type_traits/is_integral.hpp>
|
|
#include <boost/type_traits/remove_cv.hpp>
|
|
#include <boost/type_traits/remove_reference.hpp>
|
|
|
|
// Must come last.
|
|
#include <boost/iostreams/detail/config/disable_warnings.hpp>
|
|
|
|
namespace boost { namespace iostreams {
|
|
|
|
template<typename T>
|
|
void close(T& t);
|
|
|
|
template<typename T>
|
|
void close(T& t, BOOST_IOS::openmode which);
|
|
|
|
template<typename T, typename Sink>
|
|
void close(T& t, Sink& snk, BOOST_IOS::openmode which);
|
|
|
|
namespace detail {
|
|
|
|
template<typename T>
|
|
void close_all(T& t)
|
|
{
|
|
try {
|
|
boost::iostreams::close(t, BOOST_IOS::in);
|
|
} catch (...) {
|
|
try {
|
|
boost::iostreams::close(t, BOOST_IOS::out);
|
|
} catch (...) { }
|
|
throw;
|
|
}
|
|
boost::iostreams::close(t, BOOST_IOS::out);
|
|
}
|
|
|
|
template<typename T, typename Sink>
|
|
void close_all(T& t, Sink& snk)
|
|
{
|
|
try {
|
|
boost::iostreams::close(t, snk, BOOST_IOS::in);
|
|
} catch (...) {
|
|
try {
|
|
boost::iostreams::close(t, snk, BOOST_IOS::out);
|
|
} catch (...) { }
|
|
throw;
|
|
}
|
|
boost::iostreams::close(t, snk, BOOST_IOS::out);
|
|
}
|
|
|
|
} // End namespace detail.
|
|
|
|
} } // End namespaces iostreams, boost.
|
|
|
|
namespace boost { namespace iostreams {
|
|
|
|
namespace detail {
|
|
|
|
template<typename T>
|
|
struct close_impl;
|
|
|
|
} // End namespace detail.
|
|
|
|
template<typename T>
|
|
void close(T& t) { detail::close_all(t); }
|
|
|
|
template<typename T>
|
|
void close(T& t, BOOST_IOS::openmode which)
|
|
{
|
|
#ifdef BOOST_IOSTREAMS_STRICT
|
|
BOOST_ASSERT(which == BOOST_IOS::in || which == BOOST_IOS::out);
|
|
#else
|
|
if (which == (BOOST_IOS::in | BOOST_IOS::out)) {
|
|
detail::close_all(t);
|
|
return;
|
|
}
|
|
#endif
|
|
detail::close_impl<T>::close(detail::unwrap(t), which);
|
|
}
|
|
|
|
template<typename T, typename Sink>
|
|
void close(T& t, Sink& snk, BOOST_IOS::openmode which)
|
|
{
|
|
#ifdef BOOST_IOSTREAMS_STRICT
|
|
BOOST_ASSERT(which == BOOST_IOS::in || which == BOOST_IOS::out);
|
|
#else
|
|
if (which == (BOOST_IOS::in | BOOST_IOS::out)) {
|
|
detail::close_all(t, snk);
|
|
return;
|
|
}
|
|
#endif
|
|
detail::close_impl<T>::close(detail::unwrap(t), snk, which);
|
|
}
|
|
|
|
namespace detail {
|
|
|
|
//------------------Definition of close_impl----------------------------------//
|
|
|
|
struct close_boost_stream { };
|
|
struct close_filtering_stream { };
|
|
|
|
template<typename T>
|
|
struct close_tag {
|
|
typedef typename category_of<T>::type category;
|
|
typedef typename detail::unwrapped_type<T>::type unwrapped;
|
|
typedef typename
|
|
iostreams::select<
|
|
mpl::not_< is_convertible<category, closable_tag> >,
|
|
any_tag,
|
|
mpl::or_<
|
|
is_boost_stream<unwrapped>,
|
|
is_boost_stream_buffer<unwrapped>
|
|
>,
|
|
close_boost_stream,
|
|
mpl::or_<
|
|
is_filtering_stream<unwrapped>,
|
|
is_filtering_streambuf<unwrapped>
|
|
>,
|
|
close_filtering_stream,
|
|
mpl::or_<
|
|
is_convertible<category, two_sequence>,
|
|
is_convertible<category, dual_use>
|
|
>,
|
|
two_sequence,
|
|
else_,
|
|
closable_tag
|
|
>::type type;
|
|
};
|
|
|
|
template<typename T>
|
|
struct close_impl
|
|
: mpl::if_<
|
|
is_custom<T>,
|
|
operations<T>,
|
|
close_impl<BOOST_DEDUCED_TYPENAME close_tag<T>::type>
|
|
>::type
|
|
{ };
|
|
|
|
template<>
|
|
struct close_impl<any_tag> {
|
|
template<typename T>
|
|
static void close(T& t, BOOST_IOS::openmode which)
|
|
{
|
|
if (which == BOOST_IOS::out)
|
|
iostreams::flush(t);
|
|
}
|
|
|
|
template<typename T, typename Sink>
|
|
static void close(T& t, Sink& snk, BOOST_IOS::openmode which)
|
|
{
|
|
if (which == BOOST_IOS::out) {
|
|
non_blocking_adapter<Sink> nb(snk);
|
|
iostreams::flush(t, nb);
|
|
}
|
|
}
|
|
};
|
|
|
|
template<>
|
|
struct close_impl<close_boost_stream> {
|
|
template<typename T>
|
|
static void close(T& t)
|
|
{
|
|
t.close();
|
|
}
|
|
template<typename T>
|
|
static void close(T& t, BOOST_IOS::openmode which)
|
|
{
|
|
if (which == BOOST_IOS::out)
|
|
t.close();
|
|
}
|
|
};
|
|
|
|
template<>
|
|
struct close_impl<close_filtering_stream> {
|
|
template<typename T>
|
|
static void close(T& t, BOOST_IOS::openmode which)
|
|
{
|
|
typedef typename category_of<T>::type category;
|
|
const bool in = is_convertible<category, input>::value &&
|
|
!is_convertible<category, output>::value;
|
|
if (in == (which == BOOST_IOS::in) && t.is_complete())
|
|
t.pop();
|
|
}
|
|
};
|
|
|
|
template<>
|
|
struct close_impl<closable_tag> {
|
|
template<typename T>
|
|
static void close(T& t, BOOST_IOS::openmode which)
|
|
{
|
|
typedef typename category_of<T>::type category;
|
|
const bool in = is_convertible<category, input>::value &&
|
|
!is_convertible<category, output>::value;
|
|
if (in == (which == BOOST_IOS::in))
|
|
t.close();
|
|
}
|
|
template<typename T, typename Sink>
|
|
static void close(T& t, Sink& snk, BOOST_IOS::openmode which)
|
|
{
|
|
typedef typename category_of<T>::type category;
|
|
const bool in = is_convertible<category, input>::value &&
|
|
!is_convertible<category, output>::value;
|
|
if (in == (which == BOOST_IOS::in)) {
|
|
non_blocking_adapter<Sink> nb(snk);
|
|
t.close(nb);
|
|
}
|
|
}
|
|
};
|
|
|
|
template<>
|
|
struct close_impl<two_sequence> {
|
|
template<typename T>
|
|
static void close(T& t, BOOST_IOS::openmode which) { t.close(which); }
|
|
template<typename T, typename Sink>
|
|
static void close(T& t, Sink& snk, BOOST_IOS::openmode which)
|
|
{
|
|
non_blocking_adapter<Sink> nb(snk);
|
|
t.close(nb, which);
|
|
}
|
|
};
|
|
|
|
} // End namespace detail.
|
|
|
|
} } // End namespaces iostreams, boost.
|
|
|
|
#include <boost/iostreams/detail/config/enable_warnings.hpp>
|
|
|
|
#endif // #ifndef BOOST_IOSTREAMS_CLOSE_HPP_INCLUDED
|