1 /*
2  *
3  * Copyright (c) 1998-2002
4  * John Maddock
5  *
6  * Use, modification and distribution are subject to the
7  * Boost Software License, Version 1.0. (See accompanying file
8  * LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
9  *
10  */
11 
12  /*
13   *   LOCATION:    see http://www.boost.org for most recent version.
14   *   FILE         sub_match.cpp
15   *   VERSION      see <boost/version.hpp>
16   *   DESCRIPTION: Declares template class sub_match.
17   */
18 
19 #ifndef BOOST_REGEX_V4_SUB_MATCH_HPP
20 #define BOOST_REGEX_V4_SUB_MATCH_HPP
21 
22 #ifdef BOOST_MSVC
23 #pragma warning(push)
24 #pragma warning(disable: 4103)
25 #endif
26 #ifdef BOOST_HAS_ABI_HEADERS
27 #  include BOOST_ABI_PREFIX
28 #endif
29 #ifdef BOOST_MSVC
30 #pragma warning(pop)
31 #endif
32 
33 namespace boost{
34 
35 template <class BidiIterator>
36 struct sub_match : public std::pair<BidiIterator, BidiIterator>
37 {
38    typedef typename re_detail::regex_iterator_traits<BidiIterator>::value_type       value_type;
39 #if defined(BOOST_NO_STD_ITERATOR_TRAITS)
40    typedef          std::ptrdiff_t                                                   difference_type;
41 #else
42    typedef typename re_detail::regex_iterator_traits<BidiIterator>::difference_type  difference_type;
43 #endif
44    typedef          BidiIterator                                                     iterator_type;
45    typedef          BidiIterator                                                     iterator;
46    typedef          BidiIterator                                                     const_iterator;
47 
48    bool matched;
49 
sub_matchboost::sub_match50    sub_match() : std::pair<BidiIterator, BidiIterator>(), matched(false) {}
sub_matchboost::sub_match51    sub_match(BidiIterator i) : std::pair<BidiIterator, BidiIterator>(i, i), matched(false) {}
52 #if !defined(BOOST_NO_TEMPLATED_ITERATOR_CONSTRUCTORS)\
53                && !BOOST_WORKAROUND(__BORLANDC__, <= 0x0551)\
54                && !BOOST_WORKAROUND(__DECCXX_VER, BOOST_TESTED_AT(60590042))
55    template <class T, class A>
operator std::basic_string<value_type,T,A>boost::sub_match56    operator std::basic_string<value_type, T, A> ()const
57    {
58       return matched ? std::basic_string<value_type, T, A>(this->first, this->second) : std::basic_string<value_type, T, A>();
59    }
60 #else
operator std::basic_string<value_type>boost::sub_match61    operator std::basic_string<value_type> ()const
62    {
63       return str();
64    }
65 #endif
lengthboost::sub_match66    difference_type BOOST_REGEX_CALL length()const
67    {
68       difference_type n = matched ? ::boost::re_detail::distance((BidiIterator)this->first, (BidiIterator)this->second) : 0;
69       return n;
70    }
strboost::sub_match71    std::basic_string<value_type> str()const
72    {
73       std::basic_string<value_type> result;
74       if(matched)
75       {
76          std::size_t len = ::boost::re_detail::distance((BidiIterator)this->first, (BidiIterator)this->second);
77          result.reserve(len);
78          BidiIterator i = this->first;
79          while(i != this->second)
80          {
81             result.append(1, *i);
82             ++i;
83          }
84       }
85       return result;
86    }
compareboost::sub_match87    int compare(const sub_match& s)const
88    {
89       if(matched != s.matched)
90          return static_cast<int>(matched) - static_cast<int>(s.matched);
91       return str().compare(s.str());
92    }
compareboost::sub_match93    int compare(const std::basic_string<value_type>& s)const
94    {
95       return str().compare(s);
96    }
compareboost::sub_match97    int compare(const value_type* p)const
98    {
99       return str().compare(p);
100    }
101 
operator ==boost::sub_match102    bool operator==(const sub_match& that)const
103    { return compare(that) == 0; }
operator !=boost::sub_match104    bool BOOST_REGEX_CALL operator !=(const sub_match& that)const
105    { return compare(that) != 0; }
operator <boost::sub_match106    bool operator<(const sub_match& that)const
107    { return compare(that) < 0; }
operator >boost::sub_match108    bool operator>(const sub_match& that)const
109    { return compare(that) > 0; }
operator <=boost::sub_match110    bool operator<=(const sub_match& that)const
111    { return compare(that) <= 0; }
operator >=boost::sub_match112    bool operator>=(const sub_match& that)const
113    { return compare(that) >= 0; }
114 
115 #ifdef BOOST_REGEX_MATCH_EXTRA
116    typedef std::vector<sub_match<BidiIterator> > capture_sequence_type;
117 
capturesboost::sub_match118    const capture_sequence_type& captures()const
119    {
120       if(!m_captures)
121          m_captures.reset(new capture_sequence_type());
122       return *m_captures;
123    }
124    //
125    // Private implementation API: DO NOT USE!
126    //
get_capturesboost::sub_match127    capture_sequence_type& get_captures()const
128    {
129       if(!m_captures)
130          m_captures.reset(new capture_sequence_type());
131       return *m_captures;
132    }
133 
134 private:
135    mutable boost::scoped_ptr<capture_sequence_type> m_captures;
136 public:
137 
138 #endif
sub_matchboost::sub_match139    sub_match(const sub_match& that, bool
140 #ifdef BOOST_REGEX_MATCH_EXTRA
141       deep_copy
142 #endif
143       = true
144       )
145       : std::pair<BidiIterator, BidiIterator>(that),
146         matched(that.matched)
147    {
148 #ifdef BOOST_REGEX_MATCH_EXTRA
149       if(that.m_captures)
150          if(deep_copy)
151             m_captures.reset(new capture_sequence_type(*(that.m_captures)));
152 #endif
153    }
operator =boost::sub_match154    sub_match& operator=(const sub_match& that)
155    {
156       this->first = that.first;
157       this->second = that.second;
158       matched = that.matched;
159 #ifdef BOOST_REGEX_MATCH_EXTRA
160       if(that.m_captures)
161          get_captures() = *(that.m_captures);
162 #endif
163       return *this;
164    }
165 
166 
167 #ifdef BOOST_OLD_REGEX_H
168    //
169    // the following are deprecated, do not use!!
170    //
171    operator int()const;
172    operator unsigned int()const;
operator shortboost::sub_match173    operator short()const
174    {
175       return (short)(int)(*this);
176    }
operator unsigned shortboost::sub_match177    operator unsigned short()const
178    {
179       return (unsigned short)(unsigned int)(*this);
180    }
181 #endif
182 };
183 
184 typedef sub_match<const char*> csub_match;
185 typedef sub_match<std::string::const_iterator> ssub_match;
186 #ifndef BOOST_NO_WREGEX
187 typedef sub_match<const wchar_t*> wcsub_match;
188 typedef sub_match<std::wstring::const_iterator> wssub_match;
189 #endif
190 
191 // comparison to std::basic_string<> part 1:
192 template <class RandomAccessIterator, class traits, class Allocator>
operator ==(const std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type,traits,Allocator> & s,const sub_match<RandomAccessIterator> & m)193 inline bool operator == (const std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type, traits, Allocator>& s,
194                   const sub_match<RandomAccessIterator>& m)
195 { return s.compare(m.str()) == 0; }
196 template <class RandomAccessIterator, class traits, class Allocator>
operator !=(const std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type,traits,Allocator> & s,const sub_match<RandomAccessIterator> & m)197 inline bool operator != (const std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type, traits, Allocator>& s,
198                   const sub_match<RandomAccessIterator>& m)
199 { return s.compare(m.str()) != 0; }
200 template <class RandomAccessIterator, class traits, class Allocator>
operator <(const std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type,traits,Allocator> & s,const sub_match<RandomAccessIterator> & m)201 inline bool operator < (const std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type, traits, Allocator>& s,
202                  const sub_match<RandomAccessIterator>& m)
203 { return s.compare(m.str()) < 0; }
204 template <class RandomAccessIterator, class traits, class Allocator>
operator <=(const std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type,traits,Allocator> & s,const sub_match<RandomAccessIterator> & m)205 inline bool operator <= (const std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type, traits, Allocator>& s,
206                   const sub_match<RandomAccessIterator>& m)
207 { return s.compare(m.str()) <= 0; }
208 template <class RandomAccessIterator, class traits, class Allocator>
operator >=(const std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type,traits,Allocator> & s,const sub_match<RandomAccessIterator> & m)209 inline bool operator >= (const std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type, traits, Allocator>& s,
210                   const sub_match<RandomAccessIterator>& m)
211 { return s.compare(m.str()) >= 0; }
212 template <class RandomAccessIterator, class traits, class Allocator>
operator >(const std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type,traits,Allocator> & s,const sub_match<RandomAccessIterator> & m)213 inline bool operator > (const std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type, traits, Allocator>& s,
214                  const sub_match<RandomAccessIterator>& m)
215 { return s.compare(m.str()) > 0; }
216 // comparison to std::basic_string<> part 2:
217 template <class RandomAccessIterator, class traits, class Allocator>
operator ==(const sub_match<RandomAccessIterator> & m,const std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type,traits,Allocator> & s)218 inline bool operator == (const sub_match<RandomAccessIterator>& m,
219                   const std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type, traits, Allocator>& s)
220 { return m.str().compare(s) == 0; }
221 template <class RandomAccessIterator, class traits, class Allocator>
operator !=(const sub_match<RandomAccessIterator> & m,const std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type,traits,Allocator> & s)222 inline bool operator != (const sub_match<RandomAccessIterator>& m,
223                   const std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type, traits, Allocator>& s)
224 { return m.str().compare(s) != 0; }
225 template <class RandomAccessIterator, class traits, class Allocator>
operator <(const sub_match<RandomAccessIterator> & m,const std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type,traits,Allocator> & s)226 inline bool operator < (const sub_match<RandomAccessIterator>& m,
227                   const std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type, traits, Allocator>& s)
228 { return m.str().compare(s) < 0; }
229 template <class RandomAccessIterator, class traits, class Allocator>
operator >(const sub_match<RandomAccessIterator> & m,const std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type,traits,Allocator> & s)230 inline bool operator > (const sub_match<RandomAccessIterator>& m,
231                   const std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type, traits, Allocator>& s)
232 { return m.str().compare(s) > 0; }
233 template <class RandomAccessIterator, class traits, class Allocator>
operator <=(const sub_match<RandomAccessIterator> & m,const std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type,traits,Allocator> & s)234 inline bool operator <= (const sub_match<RandomAccessIterator>& m,
235                   const std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type, traits, Allocator>& s)
236 { return m.str().compare(s) <= 0; }
237 template <class RandomAccessIterator, class traits, class Allocator>
operator >=(const sub_match<RandomAccessIterator> & m,const std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type,traits,Allocator> & s)238 inline bool operator >= (const sub_match<RandomAccessIterator>& m,
239                   const std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type, traits, Allocator>& s)
240 { return m.str().compare(s) >= 0; }
241 // comparison to const charT* part 1:
242 template <class RandomAccessIterator>
operator ==(const sub_match<RandomAccessIterator> & m,typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const * s)243 inline bool operator == (const sub_match<RandomAccessIterator>& m,
244                   typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const* s)
245 { return m.str().compare(s) == 0; }
246 template <class RandomAccessIterator>
operator !=(const sub_match<RandomAccessIterator> & m,typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const * s)247 inline bool operator != (const sub_match<RandomAccessIterator>& m,
248                   typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const* s)
249 { return m.str().compare(s) != 0; }
250 template <class RandomAccessIterator>
operator >(const sub_match<RandomAccessIterator> & m,typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const * s)251 inline bool operator > (const sub_match<RandomAccessIterator>& m,
252                   typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const* s)
253 { return m.str().compare(s) > 0; }
254 template <class RandomAccessIterator>
operator <(const sub_match<RandomAccessIterator> & m,typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const * s)255 inline bool operator < (const sub_match<RandomAccessIterator>& m,
256                   typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const* s)
257 { return m.str().compare(s) < 0; }
258 template <class RandomAccessIterator>
operator >=(const sub_match<RandomAccessIterator> & m,typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const * s)259 inline bool operator >= (const sub_match<RandomAccessIterator>& m,
260                   typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const* s)
261 { return m.str().compare(s) >= 0; }
262 template <class RandomAccessIterator>
operator <=(const sub_match<RandomAccessIterator> & m,typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const * s)263 inline bool operator <= (const sub_match<RandomAccessIterator>& m,
264                   typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const* s)
265 { return m.str().compare(s) <= 0; }
266 // comparison to const charT* part 2:
267 template <class RandomAccessIterator>
operator ==(typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const * s,const sub_match<RandomAccessIterator> & m)268 inline bool operator == (typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const* s,
269                   const sub_match<RandomAccessIterator>& m)
270 { return m.str().compare(s) == 0; }
271 template <class RandomAccessIterator>
operator !=(typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const * s,const sub_match<RandomAccessIterator> & m)272 inline bool operator != (typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const* s,
273                   const sub_match<RandomAccessIterator>& m)
274 { return m.str().compare(s) != 0; }
275 template <class RandomAccessIterator>
operator <(typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const * s,const sub_match<RandomAccessIterator> & m)276 inline bool operator < (typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const* s,
277                   const sub_match<RandomAccessIterator>& m)
278 { return m.str().compare(s) > 0; }
279 template <class RandomAccessIterator>
operator >(typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const * s,const sub_match<RandomAccessIterator> & m)280 inline bool operator > (typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const* s,
281                   const sub_match<RandomAccessIterator>& m)
282 { return m.str().compare(s) < 0; }
283 template <class RandomAccessIterator>
operator <=(typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const * s,const sub_match<RandomAccessIterator> & m)284 inline bool operator <= (typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const* s,
285                   const sub_match<RandomAccessIterator>& m)
286 { return m.str().compare(s) >= 0; }
287 template <class RandomAccessIterator>
operator >=(typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const * s,const sub_match<RandomAccessIterator> & m)288 inline bool operator >= (typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const* s,
289                   const sub_match<RandomAccessIterator>& m)
290 { return m.str().compare(s) <= 0; }
291 
292 // comparison to const charT& part 1:
293 template <class RandomAccessIterator>
operator ==(const sub_match<RandomAccessIterator> & m,typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const & s)294 inline bool operator == (const sub_match<RandomAccessIterator>& m,
295                   typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const& s)
296 { return m.str().compare(0, m.length(), &s, 1) == 0; }
297 template <class RandomAccessIterator>
operator !=(const sub_match<RandomAccessIterator> & m,typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const & s)298 inline bool operator != (const sub_match<RandomAccessIterator>& m,
299                   typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const& s)
300 { return m.str().compare(0, m.length(), &s, 1) != 0; }
301 template <class RandomAccessIterator>
operator >(const sub_match<RandomAccessIterator> & m,typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const & s)302 inline bool operator > (const sub_match<RandomAccessIterator>& m,
303                   typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const& s)
304 { return m.str().compare(0, m.length(), &s, 1) > 0; }
305 template <class RandomAccessIterator>
operator <(const sub_match<RandomAccessIterator> & m,typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const & s)306 inline bool operator < (const sub_match<RandomAccessIterator>& m,
307                   typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const& s)
308 { return m.str().compare(0, m.length(), &s, 1) < 0; }
309 template <class RandomAccessIterator>
operator >=(const sub_match<RandomAccessIterator> & m,typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const & s)310 inline bool operator >= (const sub_match<RandomAccessIterator>& m,
311                   typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const& s)
312 { return m.str().compare(0, m.length(), &s, 1) >= 0; }
313 template <class RandomAccessIterator>
operator <=(const sub_match<RandomAccessIterator> & m,typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const & s)314 inline bool operator <= (const sub_match<RandomAccessIterator>& m,
315                   typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const& s)
316 { return m.str().compare(0, m.length(), &s, 1) <= 0; }
317 // comparison to const charT* part 2:
318 template <class RandomAccessIterator>
operator ==(typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const & s,const sub_match<RandomAccessIterator> & m)319 inline bool operator == (typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const& s,
320                   const sub_match<RandomAccessIterator>& m)
321 { return m.str().compare(0, m.length(), &s, 1) == 0; }
322 template <class RandomAccessIterator>
operator !=(typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const & s,const sub_match<RandomAccessIterator> & m)323 inline bool operator != (typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const& s,
324                   const sub_match<RandomAccessIterator>& m)
325 { return m.str().compare(0, m.length(), &s, 1) != 0; }
326 template <class RandomAccessIterator>
operator <(typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const & s,const sub_match<RandomAccessIterator> & m)327 inline bool operator < (typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const& s,
328                   const sub_match<RandomAccessIterator>& m)
329 { return m.str().compare(0, m.length(), &s, 1) > 0; }
330 template <class RandomAccessIterator>
operator >(typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const & s,const sub_match<RandomAccessIterator> & m)331 inline bool operator > (typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const& s,
332                   const sub_match<RandomAccessIterator>& m)
333 { return m.str().compare(0, m.length(), &s, 1) < 0; }
334 template <class RandomAccessIterator>
operator <=(typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const & s,const sub_match<RandomAccessIterator> & m)335 inline bool operator <= (typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const& s,
336                   const sub_match<RandomAccessIterator>& m)
337 { return m.str().compare(0, m.length(), &s, 1) >= 0; }
338 template <class RandomAccessIterator>
operator >=(typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const & s,const sub_match<RandomAccessIterator> & m)339 inline bool operator >= (typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const& s,
340                   const sub_match<RandomAccessIterator>& m)
341 { return m.str().compare(0, m.length(), &s, 1) <= 0; }
342 
343 // addition operators:
344 template <class RandomAccessIterator, class traits, class Allocator>
345 inline std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type, traits, Allocator>
operator +(const std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type,traits,Allocator> & s,const sub_match<RandomAccessIterator> & m)346 operator + (const std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type, traits, Allocator>& s,
347                   const sub_match<RandomAccessIterator>& m)
348 {
349    std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type, traits, Allocator> result;
350    result.reserve(s.size() + m.length() + 1);
351    return result.append(s).append(m.first, m.second);
352 }
353 template <class RandomAccessIterator, class traits, class Allocator>
354 inline std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type, traits, Allocator>
operator +(const sub_match<RandomAccessIterator> & m,const std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type,traits,Allocator> & s)355 operator + (const sub_match<RandomAccessIterator>& m,
356             const std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type, traits, Allocator>& s)
357 {
358    std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type, traits, Allocator> result;
359    result.reserve(s.size() + m.length() + 1);
360    return result.append(m.first, m.second).append(s);
361 }
362 #if !(defined(__GNUC__) && defined(BOOST_NO_STD_LOCALE))
363 template <class RandomAccessIterator>
364 inline std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type>
operator +(typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const * s,const sub_match<RandomAccessIterator> & m)365 operator + (typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const* s,
366                   const sub_match<RandomAccessIterator>& m)
367 {
368    std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type> result;
369    result.reserve(std::char_traits<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type>::length(s) + m.length() + 1);
370    return result.append(s).append(m.first, m.second);
371 }
372 template <class RandomAccessIterator>
373 inline std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type>
operator +(const sub_match<RandomAccessIterator> & m,typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const * s)374 operator + (const sub_match<RandomAccessIterator>& m,
375             typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const * s)
376 {
377    std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type> result;
378    result.reserve(std::char_traits<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type>::length(s) + m.length() + 1);
379    return result.append(m.first, m.second).append(s);
380 }
381 #else
382 // worwaround versions:
383 template <class RandomAccessIterator>
384 inline std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type>
operator +(typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const * s,const sub_match<RandomAccessIterator> & m)385 operator + (typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const* s,
386                   const sub_match<RandomAccessIterator>& m)
387 {
388    return s + m.str();
389 }
390 template <class RandomAccessIterator>
391 inline std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type>
operator +(const sub_match<RandomAccessIterator> & m,typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const * s)392 operator + (const sub_match<RandomAccessIterator>& m,
393             typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const * s)
394 {
395    return m.str() + s;
396 }
397 #endif
398 template <class RandomAccessIterator>
399 inline std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type>
operator +(typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const & s,const sub_match<RandomAccessIterator> & m)400 operator + (typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const& s,
401                   const sub_match<RandomAccessIterator>& m)
402 {
403    std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type> result;
404    result.reserve(m.length() + 2);
405    return result.append(1, s).append(m.first, m.second);
406 }
407 template <class RandomAccessIterator>
408 inline std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type>
operator +(const sub_match<RandomAccessIterator> & m,typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const & s)409 operator + (const sub_match<RandomAccessIterator>& m,
410             typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type const& s)
411 {
412    std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type> result;
413    result.reserve(m.length() + 2);
414    return result.append(m.first, m.second).append(1, s);
415 }
416 template <class RandomAccessIterator>
417 inline std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type>
operator +(const sub_match<RandomAccessIterator> & m1,const sub_match<RandomAccessIterator> & m2)418 operator + (const sub_match<RandomAccessIterator>& m1,
419             const sub_match<RandomAccessIterator>& m2)
420 {
421    std::basic_string<typename re_detail::regex_iterator_traits<RandomAccessIterator>::value_type> result;
422    result.reserve(m1.length() + m2.length() + 1);
423    return result.append(m1.first, m1.second).append(m2.first, m2.second);
424 }
425 #ifndef BOOST_NO_STD_LOCALE
426 template <class charT, class traits, class RandomAccessIterator>
427 std::basic_ostream<charT, traits>&
operator <<(std::basic_ostream<charT,traits> & os,const sub_match<RandomAccessIterator> & s)428    operator << (std::basic_ostream<charT, traits>& os,
429                 const sub_match<RandomAccessIterator>& s)
430 {
431    return (os << s.str());
432 }
433 #else
434 template <class RandomAccessIterator>
operator <<(std::ostream & os,const sub_match<RandomAccessIterator> & s)435 std::ostream& operator << (std::ostream& os,
436                            const sub_match<RandomAccessIterator>& s)
437 {
438    return (os << s.str());
439 }
440 #endif
441 
442 #ifdef BOOST_OLD_REGEX_H
443 namespace re_detail{
444 template <class BidiIterator, class charT>
do_toi(BidiIterator i,BidiIterator j,char c,int radix)445 int do_toi(BidiIterator i, BidiIterator j, char c, int radix)
446 {
447    std::string s(i, j);
448    char* p;
449    int result = std::strtol(s.c_str(), &p, radix);
450    if(*p)raise_regex_exception("Bad sub-expression");
451    return result;
452 }
453 
454 //
455 // helper:
456 template <class I, class charT>
do_toi(I & i,I j,charT c)457 int do_toi(I& i, I j, charT c)
458 {
459    int result = 0;
460    while((i != j) && (isdigit(*i)))
461    {
462       result = result*10 + (*i - '0');
463       ++i;
464    }
465    return result;
466 }
467 }
468 
469 
470 template <class BidiIterator>
operator int() const471 sub_match<BidiIterator>::operator int()const
472 {
473    BidiIterator i = first;
474    BidiIterator j = second;
475    if(i == j)raise_regex_exception("Bad sub-expression");
476    int neg = 1;
477    if((i != j) && (*i == '-'))
478    {
479       neg = -1;
480       ++i;
481    }
482    neg *= re_detail::do_toi(i, j, *i);
483    if(i != j)raise_regex_exception("Bad sub-expression");
484    return neg;
485 }
486 template <class BidiIterator>
operator unsigned int() const487 sub_match<BidiIterator>::operator unsigned int()const
488 {
489    BidiIterator i = first;
490    BidiIterator j = second;
491    if(i == j)
492       raise_regex_exception("Bad sub-expression");
493    return re_detail::do_toi(i, j, *first);
494 }
495 #endif
496 
497 } // namespace boost
498 
499 #ifdef BOOST_MSVC
500 #pragma warning(push)
501 #pragma warning(disable: 4103)
502 #endif
503 #ifdef BOOST_HAS_ABI_HEADERS
504 #  include BOOST_ABI_SUFFIX
505 #endif
506 #ifdef BOOST_MSVC
507 #pragma warning(pop)
508 #endif
509 
510 #endif
511 
512