| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731 | //  (C) Copyright Gennadiy Rozental 2004-2008.//  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/test for the library home page.////  File        : $RCSfile$////  Version     : $Revision: 57992 $////  Description : class basic_cstring wraps C string and provide std_string like //                interface// ***************************************************************************#ifndef BOOST_TEST_BASIC_CSTRING_HPP_071894GER#define BOOST_TEST_BASIC_CSTRING_HPP_071894GER// Boost.Test#include <boost/test/utils/basic_cstring/basic_cstring_fwd.hpp>#include <boost/test/utils/basic_cstring/bcs_char_traits.hpp>// STL#include <string>#include <boost/test/detail/suppress_warnings.hpp>//____________________________________________________________________________//namespace boost {namespace unit_test {// ************************************************************************** //// **************                basic_cstring                 ************** //// ************************************************************************** //template<typename CharT>class basic_cstring {    typedef basic_cstring<CharT>                        self_type;public:    // Subtypes    typedef ut_detail::bcs_char_traits<CharT>           traits_type;    typedef typename ut_detail::bcs_char_traits<CharT>::std_string  std_string;    typedef CharT                                       value_type;    typedef value_type*                                 pointer;    typedef value_type const*                           const_pointer;    typedef value_type&                                 reference;    typedef const value_type&                           const_reference;    typedef std::size_t                                 size_type;    typedef std::ptrdiff_t                              difference_type;    typedef value_type const*                           const_iterator;    typedef value_type*                                 iterator;    // !! should also present reverse_iterator, const_reverse_iterator#if !BOOST_WORKAROUND(__IBMCPP__, BOOST_TESTED_AT(600))    enum npos_type { npos = static_cast<size_type>(-1) };#else    // IBM/VisualAge version 6 is not able to handle enums larger than 4 bytes.    // But size_type is 8 bytes in 64bit mode.    static const size_type npos = -1 ;#endif    static pointer  null_str();    // Constructors; default copy constructor is generated by compiler    basic_cstring();    basic_cstring( std_string const& s );    basic_cstring( pointer s );    basic_cstring( pointer s, size_type arg_size );    basic_cstring( pointer first, pointer last );    // data access methods    value_type      operator[]( size_type index ) const;    value_type      at( size_type index ) const;    // size operators    size_type       size() const;    bool            is_empty() const;    void            clear();    void            resize( size_type new_len );    // !! only for STL container conformance use is_empty instead    bool            empty() const;     // Trimming    self_type&      trim_right( size_type trim_size );    self_type&      trim_left( size_type trim_size );    self_type&      trim_right( iterator it );    self_type&      trim_left( iterator it );#ifndef __IBMCPP__    self_type&      trim_left( self_type exclusions = self_type() ) ;    self_type&      trim_right( self_type exclusions = self_type() ) ;    self_type&      trim( self_type exclusions = self_type() ) ;#else    // VisualAge version 6 has in this case a problem with the default arguments.    self_type&      trim_left( self_type exclusions ) ;    self_type&      trim_right( self_type exclusions ) ;    self_type&      trim( self_type exclusions ) ;    self_type&      trim_left() { trim_left( self_type() ) ; }    self_type&      trim_right() { trim_right( self_type() ) ; }    self_type&      trim() { trim( self_type() ) ; }#endif    // Assignment operators    basic_cstring&  operator=( self_type const& s );    basic_cstring&  operator=( std_string const& s );    basic_cstring&  operator=( pointer s );    template<typename CharT2>    basic_cstring&  assign( basic_cstring<CharT2> const& s ) { *this = basic_cstring<CharT>( s.begin(), s.end() ); return *this; }    basic_cstring&  assign( self_type const& s, size_type pos, size_type len );    basic_cstring&  assign( std_string const& s );    basic_cstring&  assign( std_string const& s, size_type pos, size_type len );    basic_cstring&  assign( pointer s );    basic_cstring&  assign( pointer s, size_type len );    basic_cstring&  assign( pointer f, pointer l );    // swapping    void            swap( self_type& s );    // Iterators    iterator        begin();    const_iterator  begin() const;    iterator        end();    const_iterator  end() const;    // !! should have rbegin, rend    // substring search operation    size_type       find( basic_cstring ) const;    size_type       rfind( basic_cstring ) const;    self_type       substr( size_type beg_index, size_type end_index = npos ) const;private:    static self_type default_trim_ex();    // Data members    iterator        m_begin;    iterator        m_end;};//____________________________________________________________________________//template<typename CharT>inline typename basic_cstring<CharT>::pointerbasic_cstring<CharT>::null_str(){    static CharT null = 0;    return &null;}//____________________________________________________________________________//template<typename CharT>inlinebasic_cstring<CharT>::basic_cstring(): m_begin( null_str() ), m_end( m_begin ){}//____________________________________________________________________________//template<typename CharT>inlinebasic_cstring<CharT>::basic_cstring( std_string const& s ): m_begin( s.c_str() ), m_end( m_begin + s.size() ){}//____________________________________________________________________________//template<typename CharT>inlinebasic_cstring<CharT>::basic_cstring( pointer s ): m_begin( s ? s : null_str() ), m_end  ( m_begin + (s ? traits_type::length( s ) : 0 ) ){}//____________________________________________________________________________//template<typename CharT>inlinebasic_cstring<CharT>::basic_cstring( pointer s, size_type arg_size ): m_begin( s ), m_end( m_begin + arg_size ){}//____________________________________________________________________________//template<typename CharT>inlinebasic_cstring<CharT>::basic_cstring( pointer first, pointer last ): m_begin( first ), m_end( last ){}//____________________________________________________________________________//template<typename CharT>inline typename basic_cstring<CharT>::value_typebasic_cstring<CharT>::operator[]( size_type index ) const{    return m_begin[index];}//____________________________________________________________________________//template<typename CharT>inline typename basic_cstring<CharT>::value_typebasic_cstring<CharT>::at( size_type index ) const{    if( m_begin + index >= m_end )        return static_cast<value_type>(0);    return m_begin[index];}//____________________________________________________________________________//template<typename CharT>inline typename basic_cstring<CharT>::size_typebasic_cstring<CharT>::size() const{    return m_end - m_begin;}//____________________________________________________________________________//template<typename CharT>inline boolbasic_cstring<CharT>::is_empty() const{    return m_end == m_begin;}//____________________________________________________________________________//template<typename CharT>inline boolbasic_cstring<CharT>::empty() const{    return is_empty();}//____________________________________________________________________________//template<typename CharT>inline voidbasic_cstring<CharT>::clear(){    m_begin = m_end;}//____________________________________________________________________________//template<typename CharT>inline voidbasic_cstring<CharT>::resize( size_type new_len ){    if( m_begin + new_len < m_end )        m_end = m_begin + new_len;}//____________________________________________________________________________//template<typename CharT>inline basic_cstring<CharT>&basic_cstring<CharT>::trim_left( size_type trim_size ){    m_begin += trim_size;    if( m_end <= m_begin )        clear();    return *this;}//____________________________________________________________________________//template<typename CharT>inline basic_cstring<CharT>&basic_cstring<CharT>::trim_left( iterator it ){    m_begin = it;    if( m_end <= m_begin )        clear();    return *this;}//____________________________________________________________________________//template<typename CharT>inline basic_cstring<CharT>&basic_cstring<CharT>::trim_left( basic_cstring exclusions ){    if( exclusions.is_empty() )        exclusions = default_trim_ex();    iterator it;    for( it = begin(); it != end(); ++it ) {        if( traits_type::find( exclusions.begin(), exclusions.size(), *it ) == reinterpret_cast<pointer>(0) )            break;    }        return trim_left( it );}//____________________________________________________________________________//template<typename CharT>inline basic_cstring<CharT>&basic_cstring<CharT>::trim_right( size_type trim_size ){    m_end  -= trim_size;    if( m_end <= m_begin )        clear();    return *this;}//____________________________________________________________________________//template<typename CharT>inline basic_cstring<CharT>&basic_cstring<CharT>::trim_right( iterator it ){    m_end = it;    if( m_end <= m_begin )        clear();    return *this;}//____________________________________________________________________________//template<typename CharT>inline basic_cstring<CharT>&basic_cstring<CharT>::trim_right( basic_cstring exclusions ){    if( exclusions.is_empty() )        exclusions = default_trim_ex();    iterator it;    for( it = end()-1; it != begin()-1; --it ) {        if( self_type::traits_type::find( exclusions.begin(),  exclusions.size(), *it ) == reinterpret_cast<pointer>(0) )            break;    }        return trim_right( it+1 );}//____________________________________________________________________________//template<typename CharT>inline basic_cstring<CharT>&basic_cstring<CharT>::trim( basic_cstring exclusions ){    trim_left( exclusions );    trim_right( exclusions );    return *this;}//____________________________________________________________________________//template<typename CharT>inline basic_cstring<CharT>&basic_cstring<CharT>::operator=( basic_cstring<CharT> const& s ){    m_begin = s.m_begin;    m_end   = s.m_end;    return *this;}//____________________________________________________________________________//template<typename CharT>inline basic_cstring<CharT>&basic_cstring<CharT>::operator=( std_string const& s ){    return *this = self_type( s );}//____________________________________________________________________________//template<typename CharT>inline basic_cstring<CharT>&basic_cstring<CharT>::operator=( pointer s ){    return *this = self_type( s );}//____________________________________________________________________________//template<typename CharT>inline basic_cstring<CharT>&basic_cstring<CharT>::assign( basic_cstring<CharT> const& s, size_type pos, size_type len ){    return *this = self_type( s.m_begin + pos, len );}//____________________________________________________________________________//template<typename CharT>inline basic_cstring<CharT>&basic_cstring<CharT>::assign( std_string const& s ){    return *this = self_type( s );}//____________________________________________________________________________//template<typename CharT>inline basic_cstring<CharT>&basic_cstring<CharT>::assign( std_string const& s, size_type pos, size_type len ){    return *this = self_type( s.c_str() + pos, len );}//____________________________________________________________________________//template<typename CharT>inline basic_cstring<CharT>&basic_cstring<CharT>::assign( pointer s ){    return *this = self_type( s );}//____________________________________________________________________________//template<typename CharT>inline basic_cstring<CharT>&basic_cstring<CharT>::assign( pointer s, size_type len ){    return *this = self_type( s, len );}//____________________________________________________________________________//template<typename CharT>inline basic_cstring<CharT>&basic_cstring<CharT>::assign( pointer f, pointer l ){    return *this = self_type( f, l );}//____________________________________________________________________________//template<typename CharT>inline voidbasic_cstring<CharT>::swap( basic_cstring<CharT>& s ){    // do not want to include alogrithm    pointer tmp1    = m_begin;    pointer tmp2    = m_end;    m_begin         = s.m_begin;    m_end           = s.m_end;    s.m_begin       = tmp1;    s.m_end         = tmp2;}//____________________________________________________________________________//template<typename CharT>inline typename basic_cstring<CharT>::iteratorbasic_cstring<CharT>::begin(){    return m_begin;}//____________________________________________________________________________//template<typename CharT>inline typename basic_cstring<CharT>::const_iteratorbasic_cstring<CharT>::begin() const{    return m_begin;}//____________________________________________________________________________//template<typename CharT>inline typename basic_cstring<CharT>::iteratorbasic_cstring<CharT>::end(){    return m_end;}//____________________________________________________________________________//template<typename CharT>inline typename basic_cstring<CharT>::const_iteratorbasic_cstring<CharT>::end() const{    return m_end;}//____________________________________________________________________________//template<typename CharT>inline typename basic_cstring<CharT>::size_typebasic_cstring<CharT>::find( basic_cstring<CharT> str ) const{    if( str.is_empty() || str.size() > size() )        return static_cast<size_type>(npos);    const_iterator it   = begin();    const_iterator last = end() - str.size() + 1;    while( it != last ) {        if( traits_type::compare( it, str.begin(), str.size() ) == 0 )            break;        ++it;    }    return it == last ? static_cast<size_type>(npos) : it - begin();}//____________________________________________________________________________//template<typename CharT>inline typename basic_cstring<CharT>::size_typebasic_cstring<CharT>::rfind( basic_cstring<CharT> str ) const{    if( str.is_empty() || str.size() > size() )        return static_cast<size_type>(npos);    const_iterator it   = end() - str.size();    const_iterator last = begin()-1;    while( it != last ) {        if( traits_type::compare( it, str.begin(), str.size() ) == 0 )            break;        --it;    }    return it == last ? static_cast<size_type>(npos) : static_cast<size_type>(it - begin());}//____________________________________________________________________________//template<typename CharT>inline basic_cstring<CharT>basic_cstring<CharT>::substr( size_type beg_index, size_type end_index ) const{    return beg_index > size()            ?       self_type()            : end_index > size()                ?   self_type( m_begin + beg_index, m_end )                :   self_type( m_begin + beg_index, m_begin + end_index );}//____________________________________________________________________________//template<typename CharT>inline basic_cstring<CharT>basic_cstring<CharT>::default_trim_ex(){    static CharT ws[3] = { CharT(' '), CharT('\t'), CharT('\n') }; // !! wide case    return self_type( ws, 3 );}//____________________________________________________________________________//// ************************************************************************** //// **************             comparison operators             ************** //// ************************************************************************** //template<typename CharT1,typename CharT2>inline booloperator==( basic_cstring<CharT1> const& s1, basic_cstring<CharT2> const& s2 ){    typedef typename basic_cstring<CharT1>::traits_type traits_type;    return s1.size() == s2.size() &&                traits_type::compare( s1.begin(), s2.begin(), s1.size() ) == 0;}//____________________________________________________________________________//template<typename CharT1,typename CharT2>inline booloperator==( basic_cstring<CharT1> const& s1, CharT2* s2 ){#if !defined(__DMC__)    return s1 == basic_cstring<CharT2>( s2 );#else    return s1 == basic_cstring<CharT2 const>( s2 );#endif}//____________________________________________________________________________//template<typename CharT>inline booloperator==( basic_cstring<CharT> const& s1, typename basic_cstring<CharT>::std_string const& s2 ){    return s1 == basic_cstring<CharT>( s2 );}//____________________________________________________________________________//template<typename CharT1,typename CharT2>inline booloperator==( CharT1* s2, basic_cstring<CharT2> const& s1 ){    return s1 == s2;}//____________________________________________________________________________//template<typename CharT>inline booloperator==( typename basic_cstring<CharT>::std_string const& s2, basic_cstring<CharT> const& s1 ){    return s1 == s2;}//____________________________________________________________________________//template<typename CharT>inline booloperator!=( basic_cstring<CharT> const& s1, CharT* s2 ){    return !(s1 == s2);}//____________________________________________________________________________//template<typename CharT>inline booloperator!=( CharT* s2, basic_cstring<CharT> const& s1 ){    return !(s1 == s2);}//____________________________________________________________________________//template<typename CharT>inline booloperator!=( basic_cstring<CharT> const& s1, basic_cstring<CharT> const& s2 ){    return !(s1 == s2);}//____________________________________________________________________________//template<typename CharT>inline booloperator!=( basic_cstring<CharT> const& s1, typename basic_cstring<CharT>::std_string const& s2 ){    return !(s1 == s2);}//____________________________________________________________________________//template<typename CharT>inline booloperator!=( typename basic_cstring<CharT>::std_string const& s2, basic_cstring<CharT> const& s1 ){    return !(s1 == s2);}//____________________________________________________________________________//// ************************************************************************** //// **************                  first_char                  ************** //// ************************************************************************** //template<typename CharT>inline typename basic_cstring<CharT>::value_typefirst_char( basic_cstring<CharT> source ){    typedef typename basic_cstring<CharT>::value_type string_value_type;    return source.is_empty() ? static_cast<string_value_type>(0) : *source.begin();}//____________________________________________________________________________//// ************************************************************************** //// **************                  last_char                   ************** //// ************************************************************************** //template<typename CharT>inline typename basic_cstring<CharT>::value_typelast_char( basic_cstring<CharT> source ){    typedef typename basic_cstring<CharT>::value_type string_value_type;    return source.is_empty() ? static_cast<string_value_type>(0) : *(source.end()-1);}//____________________________________________________________________________//// ************************************************************************** //// **************                  assign_op                   ************** //// ************************************************************************** //template<typename CharT1, typename CharT2>inline voidassign_op( std::basic_string<CharT1>& target, basic_cstring<CharT2> src, int ){    target.assign( src.begin(), src.size() );}//____________________________________________________________________________//} // namespace unit_test} // namespace boost//____________________________________________________________________________//#include <boost/test/detail/enable_warnings.hpp>#endif // BOOST_TEST_BASIC_CSTRING_HPP_071894GER
 |