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
|