match_results.hpp 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707
  1. /*
  2. *
  3. * Copyright (c) 1998-2009
  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. * LOCATION: see http://www.boost.org for most recent version.
  13. * FILE match_results.cpp
  14. * VERSION see <boost/version.hpp>
  15. * DESCRIPTION: Declares template class match_results.
  16. */
  17. #ifndef BOOST_REGEX_V4_MATCH_RESULTS_HPP
  18. #define BOOST_REGEX_V4_MATCH_RESULTS_HPP
  19. #ifdef BOOST_MSVC
  20. #pragma warning(push)
  21. #pragma warning(disable: 4103)
  22. #endif
  23. #ifdef BOOST_HAS_ABI_HEADERS
  24. # include BOOST_ABI_PREFIX
  25. #endif
  26. #ifdef BOOST_MSVC
  27. #pragma warning(pop)
  28. #endif
  29. namespace boost{
  30. #ifdef BOOST_MSVC
  31. #pragma warning(push)
  32. #pragma warning(disable : 4251 4231)
  33. # if BOOST_MSVC < 1600
  34. # pragma warning(disable : 4660)
  35. # endif
  36. #endif
  37. namespace BOOST_REGEX_DETAIL_NS{
  38. class named_subexpressions;
  39. }
  40. template <class BidiIterator, class Allocator>
  41. class match_results
  42. {
  43. private:
  44. #ifndef BOOST_NO_STD_ALLOCATOR
  45. typedef std::vector<sub_match<BidiIterator>, Allocator> vector_type;
  46. #else
  47. typedef std::vector<sub_match<BidiIterator> > vector_type;
  48. #endif
  49. public:
  50. typedef sub_match<BidiIterator> value_type;
  51. #if !defined(BOOST_NO_STD_ALLOCATOR) && !(defined(BOOST_MSVC) && defined(_STLPORT_VERSION))
  52. typedef typename Allocator::const_reference const_reference;
  53. #else
  54. typedef const value_type& const_reference;
  55. #endif
  56. typedef const_reference reference;
  57. typedef typename vector_type::const_iterator const_iterator;
  58. typedef const_iterator iterator;
  59. typedef typename BOOST_REGEX_DETAIL_NS::regex_iterator_traits<
  60. BidiIterator>::difference_type difference_type;
  61. typedef typename Allocator::size_type size_type;
  62. typedef Allocator allocator_type;
  63. typedef typename BOOST_REGEX_DETAIL_NS::regex_iterator_traits<
  64. BidiIterator>::value_type char_type;
  65. typedef std::basic_string<char_type> string_type;
  66. typedef BOOST_REGEX_DETAIL_NS::named_subexpressions named_sub_type;
  67. // construct/copy/destroy:
  68. explicit match_results(const Allocator& a = Allocator())
  69. #ifndef BOOST_NO_STD_ALLOCATOR
  70. : m_subs(a), m_base(), m_null(), m_last_closed_paren(0), m_is_singular(true) {}
  71. #else
  72. : m_subs(), m_base(), m_null(), m_last_closed_paren(0), m_is_singular(true) { (void)a; }
  73. #endif
  74. //
  75. // IMPORTANT: in the code below, the crazy looking checks around m_is_singular are
  76. // all required because it is illegal to copy a singular iterator.
  77. // See https://svn.boost.org/trac/boost/ticket/3632.
  78. //
  79. match_results(const match_results& m)
  80. : m_subs(m.m_subs), m_named_subs(m.m_named_subs), m_last_closed_paren(m.m_last_closed_paren), m_is_singular(m.m_is_singular)
  81. {
  82. if(!m_is_singular)
  83. {
  84. m_base = m.m_base;
  85. m_null = m.m_null;
  86. }
  87. }
  88. match_results& operator=(const match_results& m)
  89. {
  90. m_subs = m.m_subs;
  91. m_named_subs = m.m_named_subs;
  92. m_last_closed_paren = m.m_last_closed_paren;
  93. m_is_singular = m.m_is_singular;
  94. if(!m_is_singular)
  95. {
  96. m_base = m.m_base;
  97. m_null = m.m_null;
  98. }
  99. return *this;
  100. }
  101. ~match_results(){}
  102. // size:
  103. size_type size() const
  104. { return empty() ? 0 : m_subs.size() - 2; }
  105. size_type max_size() const
  106. { return m_subs.max_size(); }
  107. bool empty() const
  108. { return m_subs.size() < 2; }
  109. // element access:
  110. difference_type length(int sub = 0) const
  111. {
  112. if(m_is_singular)
  113. raise_logic_error();
  114. sub += 2;
  115. if((sub < (int)m_subs.size()) && (sub > 0))
  116. return m_subs[sub].length();
  117. return 0;
  118. }
  119. difference_type length(const char_type* sub) const
  120. {
  121. if(m_is_singular)
  122. raise_logic_error();
  123. const char_type* sub_end = sub;
  124. while(*sub_end) ++sub_end;
  125. return length(named_subexpression_index(sub, sub_end));
  126. }
  127. template <class charT>
  128. difference_type length(const charT* sub) const
  129. {
  130. if(m_is_singular)
  131. raise_logic_error();
  132. const charT* sub_end = sub;
  133. while(*sub_end) ++sub_end;
  134. return length(named_subexpression_index(sub, sub_end));
  135. }
  136. template <class charT, class Traits, class A>
  137. difference_type length(const std::basic_string<charT, Traits, A>& sub) const
  138. {
  139. return length(sub.c_str());
  140. }
  141. difference_type position(size_type sub = 0) const
  142. {
  143. if(m_is_singular)
  144. raise_logic_error();
  145. sub += 2;
  146. if(sub < m_subs.size())
  147. {
  148. const sub_match<BidiIterator>& s = m_subs[sub];
  149. if(s.matched || (sub == 2))
  150. {
  151. return ::boost::BOOST_REGEX_DETAIL_NS::distance((BidiIterator)(m_base), (BidiIterator)(s.first));
  152. }
  153. }
  154. return ~static_cast<difference_type>(0);
  155. }
  156. difference_type position(const char_type* sub) const
  157. {
  158. const char_type* sub_end = sub;
  159. while(*sub_end) ++sub_end;
  160. return position(named_subexpression_index(sub, sub_end));
  161. }
  162. template <class charT>
  163. difference_type position(const charT* sub) const
  164. {
  165. const charT* sub_end = sub;
  166. while(*sub_end) ++sub_end;
  167. return position(named_subexpression_index(sub, sub_end));
  168. }
  169. template <class charT, class Traits, class A>
  170. difference_type position(const std::basic_string<charT, Traits, A>& sub) const
  171. {
  172. return position(sub.c_str());
  173. }
  174. string_type str(int sub = 0) const
  175. {
  176. if(m_is_singular)
  177. raise_logic_error();
  178. sub += 2;
  179. string_type result;
  180. if(sub < (int)m_subs.size() && (sub > 0))
  181. {
  182. const sub_match<BidiIterator>& s = m_subs[sub];
  183. if(s.matched)
  184. {
  185. result = s.str();
  186. }
  187. }
  188. return result;
  189. }
  190. string_type str(const char_type* sub) const
  191. {
  192. return (*this)[sub].str();
  193. }
  194. template <class Traits, class A>
  195. string_type str(const std::basic_string<char_type, Traits, A>& sub) const
  196. {
  197. return (*this)[sub].str();
  198. }
  199. template <class charT>
  200. string_type str(const charT* sub) const
  201. {
  202. return (*this)[sub].str();
  203. }
  204. template <class charT, class Traits, class A>
  205. string_type str(const std::basic_string<charT, Traits, A>& sub) const
  206. {
  207. return (*this)[sub].str();
  208. }
  209. const_reference operator[](int sub) const
  210. {
  211. if(m_is_singular && m_subs.empty())
  212. raise_logic_error();
  213. sub += 2;
  214. if(sub < (int)m_subs.size() && (sub >= 0))
  215. {
  216. return m_subs[sub];
  217. }
  218. return m_null;
  219. }
  220. //
  221. // Named sub-expressions:
  222. //
  223. const_reference named_subexpression(const char_type* i, const char_type* j) const
  224. {
  225. //
  226. // Scan for the leftmost *matched* subexpression with the specified named:
  227. //
  228. if(m_is_singular)
  229. raise_logic_error();
  230. BOOST_REGEX_DETAIL_NS::named_subexpressions::range_type r = m_named_subs->equal_range(i, j);
  231. while((r.first != r.second) && ((*this)[r.first->index].matched == false))
  232. ++r.first;
  233. return r.first != r.second ? (*this)[r.first->index] : m_null;
  234. }
  235. template <class charT>
  236. const_reference named_subexpression(const charT* i, const charT* j) const
  237. {
  238. BOOST_STATIC_ASSERT(sizeof(charT) <= sizeof(char_type));
  239. if(i == j)
  240. return m_null;
  241. std::vector<char_type> s;
  242. while(i != j)
  243. s.insert(s.end(), *i++);
  244. return named_subexpression(&*s.begin(), &*s.begin() + s.size());
  245. }
  246. int named_subexpression_index(const char_type* i, const char_type* j) const
  247. {
  248. //
  249. // Scan for the leftmost *matched* subexpression with the specified named.
  250. // If none found then return the leftmost expression with that name,
  251. // otherwise an invalid index:
  252. //
  253. if(m_is_singular)
  254. raise_logic_error();
  255. BOOST_REGEX_DETAIL_NS::named_subexpressions::range_type s, r;
  256. s = r = m_named_subs->equal_range(i, j);
  257. while((r.first != r.second) && ((*this)[r.first->index].matched == false))
  258. ++r.first;
  259. if(r.first == r.second)
  260. r = s;
  261. return r.first != r.second ? r.first->index : -20;
  262. }
  263. template <class charT>
  264. int named_subexpression_index(const charT* i, const charT* j) const
  265. {
  266. BOOST_STATIC_ASSERT(sizeof(charT) <= sizeof(char_type));
  267. if(i == j)
  268. return -20;
  269. std::vector<char_type> s;
  270. while(i != j)
  271. s.insert(s.end(), *i++);
  272. return named_subexpression_index(&*s.begin(), &*s.begin() + s.size());
  273. }
  274. template <class Traits, class A>
  275. const_reference operator[](const std::basic_string<char_type, Traits, A>& s) const
  276. {
  277. return named_subexpression(s.c_str(), s.c_str() + s.size());
  278. }
  279. const_reference operator[](const char_type* p) const
  280. {
  281. const char_type* e = p;
  282. while(*e) ++e;
  283. return named_subexpression(p, e);
  284. }
  285. template <class charT>
  286. const_reference operator[](const charT* p) const
  287. {
  288. BOOST_STATIC_ASSERT(sizeof(charT) <= sizeof(char_type));
  289. if(*p == 0)
  290. return m_null;
  291. std::vector<char_type> s;
  292. while(*p)
  293. s.insert(s.end(), *p++);
  294. return named_subexpression(&*s.begin(), &*s.begin() + s.size());
  295. }
  296. template <class charT, class Traits, class A>
  297. const_reference operator[](const std::basic_string<charT, Traits, A>& ns) const
  298. {
  299. BOOST_STATIC_ASSERT(sizeof(charT) <= sizeof(char_type));
  300. if(ns.empty())
  301. return m_null;
  302. std::vector<char_type> s;
  303. for(unsigned i = 0; i < ns.size(); ++i)
  304. s.insert(s.end(), ns[i]);
  305. return named_subexpression(&*s.begin(), &*s.begin() + s.size());
  306. }
  307. const_reference prefix() const
  308. {
  309. if(m_is_singular)
  310. raise_logic_error();
  311. return (*this)[-1];
  312. }
  313. const_reference suffix() const
  314. {
  315. if(m_is_singular)
  316. raise_logic_error();
  317. return (*this)[-2];
  318. }
  319. const_iterator begin() const
  320. {
  321. return (m_subs.size() > 2) ? (m_subs.begin() + 2) : m_subs.end();
  322. }
  323. const_iterator end() const
  324. {
  325. return m_subs.end();
  326. }
  327. // format:
  328. template <class OutputIterator, class Functor>
  329. OutputIterator format(OutputIterator out,
  330. Functor fmt,
  331. match_flag_type flags = format_default) const
  332. {
  333. if(m_is_singular)
  334. raise_logic_error();
  335. typedef typename BOOST_REGEX_DETAIL_NS::compute_functor_type<Functor, match_results<BidiIterator, Allocator>, OutputIterator>::type F;
  336. F func(fmt);
  337. return func(*this, out, flags);
  338. }
  339. template <class Functor>
  340. string_type format(Functor fmt, match_flag_type flags = format_default) const
  341. {
  342. if(m_is_singular)
  343. raise_logic_error();
  344. std::basic_string<char_type> result;
  345. BOOST_REGEX_DETAIL_NS::string_out_iterator<std::basic_string<char_type> > i(result);
  346. typedef typename BOOST_REGEX_DETAIL_NS::compute_functor_type<Functor, match_results<BidiIterator, Allocator>, BOOST_REGEX_DETAIL_NS::string_out_iterator<std::basic_string<char_type> > >::type F;
  347. F func(fmt);
  348. func(*this, i, flags);
  349. return result;
  350. }
  351. // format with locale:
  352. template <class OutputIterator, class Functor, class RegexT>
  353. OutputIterator format(OutputIterator out,
  354. Functor fmt,
  355. match_flag_type flags,
  356. const RegexT& re) const
  357. {
  358. if(m_is_singular)
  359. raise_logic_error();
  360. typedef ::boost::regex_traits_wrapper<typename RegexT::traits_type> traits_type;
  361. typedef typename BOOST_REGEX_DETAIL_NS::compute_functor_type<Functor, match_results<BidiIterator, Allocator>, OutputIterator, traits_type>::type F;
  362. F func(fmt);
  363. return func(*this, out, flags, re.get_traits());
  364. }
  365. template <class RegexT, class Functor>
  366. string_type format(Functor fmt,
  367. match_flag_type flags,
  368. const RegexT& re) const
  369. {
  370. if(m_is_singular)
  371. raise_logic_error();
  372. typedef ::boost::regex_traits_wrapper<typename RegexT::traits_type> traits_type;
  373. std::basic_string<char_type> result;
  374. BOOST_REGEX_DETAIL_NS::string_out_iterator<std::basic_string<char_type> > i(result);
  375. typedef typename BOOST_REGEX_DETAIL_NS::compute_functor_type<Functor, match_results<BidiIterator, Allocator>, BOOST_REGEX_DETAIL_NS::string_out_iterator<std::basic_string<char_type> >, traits_type >::type F;
  376. F func(fmt);
  377. func(*this, i, flags, re.get_traits());
  378. return result;
  379. }
  380. const_reference get_last_closed_paren()const
  381. {
  382. if(m_is_singular)
  383. raise_logic_error();
  384. return m_last_closed_paren == 0 ? m_null : (*this)[m_last_closed_paren];
  385. }
  386. allocator_type get_allocator() const
  387. {
  388. #ifndef BOOST_NO_STD_ALLOCATOR
  389. return m_subs.get_allocator();
  390. #else
  391. return allocator_type();
  392. #endif
  393. }
  394. void swap(match_results& that)
  395. {
  396. std::swap(m_subs, that.m_subs);
  397. std::swap(m_named_subs, that.m_named_subs);
  398. std::swap(m_last_closed_paren, that.m_last_closed_paren);
  399. if(m_is_singular)
  400. {
  401. if(!that.m_is_singular)
  402. {
  403. m_base = that.m_base;
  404. m_null = that.m_null;
  405. }
  406. }
  407. else if(that.m_is_singular)
  408. {
  409. that.m_base = m_base;
  410. that.m_null = m_null;
  411. }
  412. else
  413. {
  414. std::swap(m_base, that.m_base);
  415. std::swap(m_null, that.m_null);
  416. }
  417. std::swap(m_is_singular, that.m_is_singular);
  418. }
  419. bool operator==(const match_results& that)const
  420. {
  421. if(m_is_singular)
  422. {
  423. return that.m_is_singular;
  424. }
  425. else if(that.m_is_singular)
  426. {
  427. return false;
  428. }
  429. return (m_subs == that.m_subs) && (m_base == that.m_base) && (m_last_closed_paren == that.m_last_closed_paren);
  430. }
  431. bool operator!=(const match_results& that)const
  432. { return !(*this == that); }
  433. #ifdef BOOST_REGEX_MATCH_EXTRA
  434. typedef typename sub_match<BidiIterator>::capture_sequence_type capture_sequence_type;
  435. const capture_sequence_type& captures(int i)const
  436. {
  437. if(m_is_singular)
  438. raise_logic_error();
  439. return (*this)[i].captures();
  440. }
  441. #endif
  442. //
  443. // private access functions:
  444. void BOOST_REGEX_CALL set_second(BidiIterator i)
  445. {
  446. BOOST_ASSERT(m_subs.size() > 2);
  447. m_subs[2].second = i;
  448. m_subs[2].matched = true;
  449. m_subs[0].first = i;
  450. m_subs[0].matched = (m_subs[0].first != m_subs[0].second);
  451. m_null.first = i;
  452. m_null.second = i;
  453. m_null.matched = false;
  454. m_is_singular = false;
  455. }
  456. void BOOST_REGEX_CALL set_second(BidiIterator i, size_type pos, bool m = true, bool escape_k = false)
  457. {
  458. if(pos)
  459. m_last_closed_paren = static_cast<int>(pos);
  460. pos += 2;
  461. BOOST_ASSERT(m_subs.size() > pos);
  462. m_subs[pos].second = i;
  463. m_subs[pos].matched = m;
  464. if((pos == 2) && !escape_k)
  465. {
  466. m_subs[0].first = i;
  467. m_subs[0].matched = (m_subs[0].first != m_subs[0].second);
  468. m_null.first = i;
  469. m_null.second = i;
  470. m_null.matched = false;
  471. m_is_singular = false;
  472. }
  473. }
  474. void BOOST_REGEX_CALL set_size(size_type n, BidiIterator i, BidiIterator j)
  475. {
  476. value_type v(j);
  477. size_type len = m_subs.size();
  478. if(len > n + 2)
  479. {
  480. m_subs.erase(m_subs.begin()+n+2, m_subs.end());
  481. std::fill(m_subs.begin(), m_subs.end(), v);
  482. }
  483. else
  484. {
  485. std::fill(m_subs.begin(), m_subs.end(), v);
  486. if(n+2 != len)
  487. m_subs.insert(m_subs.end(), n+2-len, v);
  488. }
  489. m_subs[1].first = i;
  490. m_last_closed_paren = 0;
  491. }
  492. void BOOST_REGEX_CALL set_base(BidiIterator pos)
  493. {
  494. m_base = pos;
  495. }
  496. BidiIterator base()const
  497. {
  498. return m_base;
  499. }
  500. void BOOST_REGEX_CALL set_first(BidiIterator i)
  501. {
  502. BOOST_ASSERT(m_subs.size() > 2);
  503. // set up prefix:
  504. m_subs[1].second = i;
  505. m_subs[1].matched = (m_subs[1].first != i);
  506. // set up $0:
  507. m_subs[2].first = i;
  508. // zero out everything else:
  509. for(size_type n = 3; n < m_subs.size(); ++n)
  510. {
  511. m_subs[n].first = m_subs[n].second = m_subs[0].second;
  512. m_subs[n].matched = false;
  513. }
  514. }
  515. void BOOST_REGEX_CALL set_first(BidiIterator i, size_type pos, bool escape_k = false)
  516. {
  517. BOOST_ASSERT(pos+2 < m_subs.size());
  518. if(pos || escape_k)
  519. {
  520. m_subs[pos+2].first = i;
  521. if(escape_k)
  522. {
  523. m_subs[1].second = i;
  524. m_subs[1].matched = (m_subs[1].first != m_subs[1].second);
  525. }
  526. }
  527. else
  528. set_first(i);
  529. }
  530. void BOOST_REGEX_CALL maybe_assign(const match_results<BidiIterator, Allocator>& m);
  531. void BOOST_REGEX_CALL set_named_subs(boost::shared_ptr<named_sub_type> subs)
  532. {
  533. m_named_subs = subs;
  534. }
  535. private:
  536. //
  537. // Error handler called when an uninitialized match_results is accessed:
  538. //
  539. static void raise_logic_error()
  540. {
  541. std::logic_error e("Attempt to access an uninitialzed boost::match_results<> class.");
  542. boost::throw_exception(e);
  543. }
  544. vector_type m_subs; // subexpressions
  545. BidiIterator m_base; // where the search started from
  546. sub_match<BidiIterator> m_null; // a null match
  547. boost::shared_ptr<named_sub_type> m_named_subs; // Shared copy of named subs in the regex object
  548. int m_last_closed_paren; // Last ) to be seen - used for formatting
  549. bool m_is_singular; // True if our stored iterators are singular
  550. };
  551. template <class BidiIterator, class Allocator>
  552. void BOOST_REGEX_CALL match_results<BidiIterator, Allocator>::maybe_assign(const match_results<BidiIterator, Allocator>& m)
  553. {
  554. if(m_is_singular)
  555. {
  556. *this = m;
  557. return;
  558. }
  559. const_iterator p1, p2;
  560. p1 = begin();
  561. p2 = m.begin();
  562. //
  563. // Distances are measured from the start of *this* match, unless this isn't
  564. // a valid match in which case we use the start of the whole sequence. Note that
  565. // no subsequent match-candidate can ever be to the left of the first match found.
  566. // This ensures that when we are using bidirectional iterators, that distances
  567. // measured are as short as possible, and therefore as efficient as possible
  568. // to compute. Finally note that we don't use the "matched" data member to test
  569. // whether a sub-expression is a valid match, because partial matches set this
  570. // to false for sub-expression 0.
  571. //
  572. BidiIterator l_end = this->suffix().second;
  573. BidiIterator l_base = (p1->first == l_end) ? this->prefix().first : (*this)[0].first;
  574. difference_type len1 = 0;
  575. difference_type len2 = 0;
  576. difference_type base1 = 0;
  577. difference_type base2 = 0;
  578. std::size_t i;
  579. for(i = 0; i < size(); ++i, ++p1, ++p2)
  580. {
  581. //
  582. // Leftmost takes priority over longest; handle special cases
  583. // where distances need not be computed first (an optimisation
  584. // for bidirectional iterators: ensure that we don't accidently
  585. // compute the length of the whole sequence, as this can be really
  586. // expensive).
  587. //
  588. if(p1->first == l_end)
  589. {
  590. if(p2->first != l_end)
  591. {
  592. // p2 must be better than p1, and no need to calculate
  593. // actual distances:
  594. base1 = 1;
  595. base2 = 0;
  596. break;
  597. }
  598. else
  599. {
  600. // *p1 and *p2 are either unmatched or match end-of sequence,
  601. // either way no need to calculate distances:
  602. if((p1->matched == false) && (p2->matched == true))
  603. break;
  604. if((p1->matched == true) && (p2->matched == false))
  605. return;
  606. continue;
  607. }
  608. }
  609. else if(p2->first == l_end)
  610. {
  611. // p1 better than p2, and no need to calculate distances:
  612. return;
  613. }
  614. base1 = ::boost::BOOST_REGEX_DETAIL_NS::distance(l_base, p1->first);
  615. base2 = ::boost::BOOST_REGEX_DETAIL_NS::distance(l_base, p2->first);
  616. BOOST_ASSERT(base1 >= 0);
  617. BOOST_ASSERT(base2 >= 0);
  618. if(base1 < base2) return;
  619. if(base2 < base1) break;
  620. len1 = ::boost::BOOST_REGEX_DETAIL_NS::distance((BidiIterator)p1->first, (BidiIterator)p1->second);
  621. len2 = ::boost::BOOST_REGEX_DETAIL_NS::distance((BidiIterator)p2->first, (BidiIterator)p2->second);
  622. BOOST_ASSERT(len1 >= 0);
  623. BOOST_ASSERT(len2 >= 0);
  624. if((len1 != len2) || ((p1->matched == false) && (p2->matched == true)))
  625. break;
  626. if((p1->matched == true) && (p2->matched == false))
  627. return;
  628. }
  629. if(i == size())
  630. return;
  631. if(base2 < base1)
  632. *this = m;
  633. else if((len2 > len1) || ((p1->matched == false) && (p2->matched == true)) )
  634. *this = m;
  635. }
  636. template <class BidiIterator, class Allocator>
  637. void swap(match_results<BidiIterator, Allocator>& a, match_results<BidiIterator, Allocator>& b)
  638. {
  639. a.swap(b);
  640. }
  641. #ifndef BOOST_NO_STD_LOCALE
  642. template <class charT, class traits, class BidiIterator, class Allocator>
  643. std::basic_ostream<charT, traits>&
  644. operator << (std::basic_ostream<charT, traits>& os,
  645. const match_results<BidiIterator, Allocator>& s)
  646. {
  647. return (os << s.str());
  648. }
  649. #else
  650. template <class BidiIterator, class Allocator>
  651. std::ostream& operator << (std::ostream& os,
  652. const match_results<BidiIterator, Allocator>& s)
  653. {
  654. return (os << s.str());
  655. }
  656. #endif
  657. #ifdef BOOST_MSVC
  658. #pragma warning(pop)
  659. #endif
  660. } // namespace boost
  661. #ifdef BOOST_MSVC
  662. #pragma warning(push)
  663. #pragma warning(disable: 4103)
  664. #endif
  665. #ifdef BOOST_HAS_ABI_HEADERS
  666. # include BOOST_ABI_SUFFIX
  667. #endif
  668. #ifdef BOOST_MSVC
  669. #pragma warning(pop)
  670. #endif
  671. #endif