os/ossrv/ossrv_pub/boost_apis/boost/xpressive/basic_regex.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
///////////////////////////////////////////////////////////////////////////////
sl@0
     2
/// \file basic_regex.hpp
sl@0
     3
/// Contains the definition of the basic_regex\<\> class template and its
sl@0
     4
/// associated helper functions.
sl@0
     5
//
sl@0
     6
//  Copyright 2004 Eric Niebler. Distributed under the Boost
sl@0
     7
//  Software License, Version 1.0. (See accompanying file
sl@0
     8
//  LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
sl@0
     9
sl@0
    10
#ifndef BOOST_XPRESSIVE_BASIC_REGEX_HPP_EAN_10_04_2005
sl@0
    11
#define BOOST_XPRESSIVE_BASIC_REGEX_HPP_EAN_10_04_2005
sl@0
    12
sl@0
    13
// MS compatible compilers support #pragma once
sl@0
    14
#if defined(_MSC_VER) && (_MSC_VER >= 1020)
sl@0
    15
# pragma once
sl@0
    16
#endif
sl@0
    17
sl@0
    18
#ifdef BOOST_XPRESSIVE_DEBUG_TRACKING_POINTER
sl@0
    19
# include <iostream>
sl@0
    20
#endif
sl@0
    21
#include <boost/mpl/bool.hpp>
sl@0
    22
#include <boost/xpressive/proto/proto_fwd.hpp>
sl@0
    23
#include <boost/xpressive/regex_constants.hpp>
sl@0
    24
#include <boost/xpressive/detail/detail_fwd.hpp>
sl@0
    25
#include <boost/xpressive/detail/core/regex_impl.hpp>
sl@0
    26
sl@0
    27
namespace boost { namespace xpressive
sl@0
    28
{
sl@0
    29
sl@0
    30
///////////////////////////////////////////////////////////////////////////////
sl@0
    31
// basic_regex
sl@0
    32
//
sl@0
    33
/// \brief Class template basic_regex\<\> is a class for holding a compiled regular expression.
sl@0
    34
template<typename BidiIter>
sl@0
    35
struct basic_regex
sl@0
    36
{
sl@0
    37
    typedef BidiIter iterator_type;
sl@0
    38
    typedef typename iterator_value<BidiIter>::type char_type;
sl@0
    39
    typedef std::basic_string<char_type> string_type;
sl@0
    40
    typedef regex_constants::syntax_option_type flag_type;
sl@0
    41
sl@0
    42
    /// \post regex_id()    == 0
sl@0
    43
    /// \post mark_count()  == 0
sl@0
    44
    basic_regex()
sl@0
    45
      : impl_()
sl@0
    46
    {
sl@0
    47
    }
sl@0
    48
sl@0
    49
    /// \param that The basic_regex object to copy.
sl@0
    50
    /// \post regex_id()    == that.regex_id()
sl@0
    51
    /// \post mark_count()  == that.mark_count()
sl@0
    52
    basic_regex(basic_regex<BidiIter> const &that)
sl@0
    53
      : impl_(that.impl_)
sl@0
    54
    {
sl@0
    55
    }
sl@0
    56
sl@0
    57
    /// \param that The basic_regex object to copy.
sl@0
    58
    /// \post regex_id()    == that.regex_id()
sl@0
    59
    /// \post mark_count()  == that.mark_count()
sl@0
    60
    /// \return *this
sl@0
    61
    basic_regex<BidiIter> &operator =(basic_regex<BidiIter> const &that)
sl@0
    62
    {
sl@0
    63
        this->impl_ = that.impl_;
sl@0
    64
        return *this;
sl@0
    65
    }
sl@0
    66
sl@0
    67
    /// Construct from a static regular expression.
sl@0
    68
    ///
sl@0
    69
    /// \param  xpr The static regular expression
sl@0
    70
    /// \pre    Xpr is the type of a static regular expression.
sl@0
    71
    /// \post   regex_id()   != 0
sl@0
    72
    /// \post   mark_count() \>= 0
sl@0
    73
    template<typename Xpr>
sl@0
    74
    basic_regex(Xpr const &xpr)
sl@0
    75
      : impl_()
sl@0
    76
    {
sl@0
    77
        this->operator =(xpr);
sl@0
    78
    }
sl@0
    79
sl@0
    80
    /// Construct from a static regular expression.
sl@0
    81
    ///
sl@0
    82
    /// \param  xpr The static regular expression.
sl@0
    83
    /// \pre    Xpr is the type of a static regular expression.
sl@0
    84
    /// \post   regex_id()   != 0
sl@0
    85
    /// \post   mark_count() \>= 0
sl@0
    86
    /// \throw  std::bad_alloc on out of memory
sl@0
    87
    /// \return *this
sl@0
    88
    template<typename Xpr>
sl@0
    89
    basic_regex<BidiIter> &operator =(Xpr const &xpr)
sl@0
    90
    {
sl@0
    91
        detail::static_compile(xpr, *this->impl_.get());
sl@0
    92
        return *this;
sl@0
    93
    }
sl@0
    94
sl@0
    95
    /// Returns the count of capturing sub-expressions in this regular expression
sl@0
    96
    ///
sl@0
    97
    std::size_t mark_count() const
sl@0
    98
    {
sl@0
    99
        return this->impl_ ? this->impl_->mark_count_ : 0;
sl@0
   100
    }
sl@0
   101
sl@0
   102
    /// Returns a token which uniquely identifies this regular expression.
sl@0
   103
    ///
sl@0
   104
    regex_id_type regex_id() const
sl@0
   105
    {
sl@0
   106
        return this->impl_ ? this->impl_->xpr_.get() : 0;
sl@0
   107
    }
sl@0
   108
sl@0
   109
    /// Swaps the contents of this basic_regex object with another.
sl@0
   110
    ///
sl@0
   111
    /// \param      that The other basic_regex object.
sl@0
   112
    /// \attention  This is a shallow swap that does not do reference tracking. If you embed
sl@0
   113
    /// a basic_regex object by reference in another regular expression and then swap its
sl@0
   114
    /// contents with another basic_regex object, the change will not be visible to the enclosing
sl@0
   115
    /// regular expression. It is done this way to ensure that swap() cannot throw.
sl@0
   116
    /// \throw nothrow
sl@0
   117
    void swap(basic_regex<BidiIter> &that) // throw()
sl@0
   118
    {
sl@0
   119
        this->impl_.swap(that.impl_);
sl@0
   120
    }
sl@0
   121
sl@0
   122
    /// Factory method for building a regex object from a string.
sl@0
   123
    /// Equivalent to regex_compiler\< BidiIter \>().compile(str, flags);
sl@0
   124
    ///
sl@0
   125
    /// \param str The std::basic_string containing the regular expression.
sl@0
   126
    /// \param flags Optional bitmask of type syntax_option_type to control how str is interpreted.
sl@0
   127
    static basic_regex<BidiIter> compile(string_type const &str, flag_type flags = regex_constants::ECMAScript)
sl@0
   128
    {
sl@0
   129
        return regex_compiler<BidiIter>().compile(str, flags);
sl@0
   130
    }
sl@0
   131
sl@0
   132
    // for binding actions to this regex when it is nested statically in another regex
sl@0
   133
    /// INTERNAL ONLY
sl@0
   134
    template<typename Action>
sl@0
   135
    proto::binary_op
sl@0
   136
    <
sl@0
   137
        proto::unary_op<basic_regex<BidiIter>, proto::noop_tag>
sl@0
   138
      , proto::unary_op<Action, proto::noop_tag>
sl@0
   139
      , proto::right_shift_tag
sl@0
   140
    > const
sl@0
   141
    operator [](detail::action_matcher<Action> const &action) const
sl@0
   142
    {
sl@0
   143
        return proto::noop(*this) >> proto::noop(*static_cast<Action const *>(&action));
sl@0
   144
    }
sl@0
   145
sl@0
   146
    //{{AFX_DEBUG
sl@0
   147
    #ifdef BOOST_XPRESSIVE_DEBUG_TRACKING_POINTER
sl@0
   148
    // BUGBUG debug only
sl@0
   149
    /// INTERNAL ONLY
sl@0
   150
    friend std::ostream &operator <<(std::ostream &sout, basic_regex<BidiIter> const &rex)
sl@0
   151
    {
sl@0
   152
        rex.dump_(sout);
sl@0
   153
        return sout;
sl@0
   154
    }
sl@0
   155
    #endif
sl@0
   156
    //}}AFX_DEBUG
sl@0
   157
sl@0
   158
private:
sl@0
   159
    friend struct detail::core_access<BidiIter>;
sl@0
   160
sl@0
   161
    // Avoid a common programming mistake. Construction from a string is ambiguous. It could mean
sl@0
   162
    //   sregex rx = sregex::compile(str); // compile the string into a regex
sl@0
   163
    // or
sl@0
   164
    //   sregex rx = as_xpr(str);          // treat the string as a literal
sl@0
   165
    // Since there is no easy way to disambiguate, disallow it and force users to say what they mean
sl@0
   166
    /// INTERNAL ONLY
sl@0
   167
    basic_regex(char_type const *);
sl@0
   168
    /// INTERNAL ONLY
sl@0
   169
    basic_regex(string_type const &);
sl@0
   170
sl@0
   171
    // used from parser, via core_access
sl@0
   172
    /// INTERNAL ONLY
sl@0
   173
    explicit basic_regex(detail::regex_impl<BidiIter> const &that)
sl@0
   174
      : impl_()
sl@0
   175
    {
sl@0
   176
        this->impl_.tracking_copy(that);
sl@0
   177
    }
sl@0
   178
sl@0
   179
    /// INTERNAL ONLY
sl@0
   180
    bool match_(detail::state_type<BidiIter> &state) const
sl@0
   181
    {
sl@0
   182
        return this->impl_->xpr_->match(state);
sl@0
   183
    }
sl@0
   184
sl@0
   185
    // Returns true if this basic_regex object does not contain a valid regular expression.
sl@0
   186
    /// INTERNAL ONLY
sl@0
   187
    bool invalid_() const
sl@0
   188
    {
sl@0
   189
        return !this->impl_ || !this->impl_->xpr_;
sl@0
   190
    }
sl@0
   191
sl@0
   192
    /// INTERNAL ONLY
sl@0
   193
    void dump_(std::ostream &sout) const;
sl@0
   194
sl@0
   195
    // the tracking_ptr manages lazy-init, COW, cycle-breaking, and
sl@0
   196
    // reference/dependency tracking.
sl@0
   197
    detail::tracking_ptr<detail::regex_impl<BidiIter> > impl_;
sl@0
   198
};
sl@0
   199
sl@0
   200
//{{AFX_DEBUG
sl@0
   201
#ifdef BOOST_XPRESSIVE_DEBUG_TRACKING_POINTER
sl@0
   202
///////////////////////////////////////////////////////////////////////////////
sl@0
   203
// dump_
sl@0
   204
/// INTERNAL ONLY
sl@0
   205
template<typename BidiIter>
sl@0
   206
inline void basic_regex<BidiIter>::dump_(std::ostream &sout) const
sl@0
   207
{
sl@0
   208
    if(!this->impl_)
sl@0
   209
    {
sl@0
   210
        sout << "<null> refs={} deps={}";
sl@0
   211
    }
sl@0
   212
    else
sl@0
   213
    {
sl@0
   214
        sout << *this->impl_;
sl@0
   215
    }
sl@0
   216
}
sl@0
   217
#endif
sl@0
   218
//}}AFX_DEBUG
sl@0
   219
sl@0
   220
///////////////////////////////////////////////////////////////////////////////
sl@0
   221
// swap
sl@0
   222
/// \brief      Swaps the contents of two basic_regex objects.
sl@0
   223
/// \param      left The first basic_regex object.
sl@0
   224
/// \param      right The second basic_regex object.
sl@0
   225
/// \attention  This is a shallow swap that does not do reference tracking.
sl@0
   226
/// If you embed a basic_regex object by reference in another regular expression
sl@0
   227
/// and then swap its contents with another basic_regex object, the change will
sl@0
   228
/// not be visible to the enclosing regular expression. It is done this way to
sl@0
   229
/// ensure that swap() cannot throw.
sl@0
   230
/// \throw nothrow
sl@0
   231
template<typename BidiIter>
sl@0
   232
inline void swap(basic_regex<BidiIter> &left, basic_regex<BidiIter> &right) // throw()
sl@0
   233
{
sl@0
   234
    left.swap(right);
sl@0
   235
}
sl@0
   236
sl@0
   237
}} // namespace boost::xpressive
sl@0
   238
sl@0
   239
#endif // BOOST_XPRESSIVE_REGEX_HPP_EAN_10_04_2005