1
0
forked from Alepha/Alepha

Make the underlying streambuf private in stacks.

This commit is contained in:
2024-04-03 15:01:52 -04:00
parent 8250680e6c
commit d8689b85e8
5 changed files with 55 additions and 24 deletions

View File

@ -4,6 +4,9 @@ static_assert( __cplusplus > 2020'99 );
#include <cassert>
#include <istream>
#include <ostream>
namespace Alepha::Hydrogen::IOStreams::detail::StackableStreambuf_m
{
namespace
@ -47,13 +50,26 @@ namespace Alepha::Hydrogen::IOStreams::detail::StackableStreambuf_m
const auto *const current= dynamic_cast< StackableStreambuf * >( os.rdbuf() );
if( not current ) return false;
os.rdbuf( current->underlying );
current->unhook( os );
releaseTop( os, token );
return true;
}
inline bool
releaseTop( std::istream &is )
{
const auto *const current= dynamic_cast< StackableStreambuf * >( is.rdbuf() );
if( not current ) return false;
current->unhook( is );
releaseTop( is, token );
return true;
}
inline void
releaseStack( std::ios_base &ios )
{
@ -84,6 +100,13 @@ namespace Alepha::Hydrogen::IOStreams::detail::StackableStreambuf_m
return os;
}
std::istream &
impl::operator >> ( std::istream &is, PopStack )
{
if( not releaseTop( is ) ) throw std::logic_error( "IStream has no stacked streambufs!" );
return is;
}
StackableStreambuf::~StackableStreambuf() {}
StackableStreambuf::StackableStreambuf( std::ios &host )
@ -111,4 +134,9 @@ namespace Alepha::Hydrogen::IOStreams::detail::StackableStreambuf_m
for( std::streamsize i= 0; i< amt; ++i ) overflow( data[ i ] );
return amt;
}
void StackableStreambuf::unhook( std::ostream &os ) const { os.rdbuf( underlying ); }
void StackableStreambuf::unhook( std::istream &is ) const { is.rdbuf( underlying ); }
std::ostream &StackableStreambuf::out( std::ostream &&res ) const { res.rdbuf( underlying ); return *&res; }
}