basic_resolver_iterator.hpp 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188
  1. //
  2. // basic_resolver_iterator.hpp
  3. // ~~~~~~~~~~~~~~~~~~~~~~~~~~~
  4. //
  5. // Copyright (c) 2003-2010 Christopher M. Kohlhoff (chris at kohlhoff dot com)
  6. //
  7. // Distributed under the Boost Software License, Version 1.0. (See accompanying
  8. // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
  9. //
  10. #ifndef ASIO_IP_BASIC_RESOLVER_ITERATOR_HPP
  11. #define ASIO_IP_BASIC_RESOLVER_ITERATOR_HPP
  12. #if defined(_MSC_VER) && (_MSC_VER >= 1200)
  13. # pragma once
  14. #endif // defined(_MSC_VER) && (_MSC_VER >= 1200)
  15. #include "asio/detail/push_options.hpp"
  16. #include "asio/detail/push_options.hpp"
  17. #include <boost/iterator.hpp>
  18. #include <boost/shared_ptr.hpp>
  19. #include <cstring>
  20. #include <string>
  21. #include <vector>
  22. #include "asio/detail/pop_options.hpp"
  23. #include "asio/detail/socket_ops.hpp"
  24. #include "asio/detail/socket_types.hpp"
  25. #include "asio/ip/basic_resolver_entry.hpp"
  26. namespace asio {
  27. namespace ip {
  28. /// An iterator over the entries produced by a resolver.
  29. /**
  30. * The asio::ip::basic_resolver_iterator class template is used to define
  31. * iterators over the results returned by a resolver.
  32. *
  33. * The iterator's value_type, obtained when the iterator is dereferenced, is:
  34. * @code const basic_resolver_entry<InternetProtocol> @endcode
  35. *
  36. * @par Thread Safety
  37. * @e Distinct @e objects: Safe.@n
  38. * @e Shared @e objects: Unsafe.
  39. */
  40. template <typename InternetProtocol>
  41. class basic_resolver_iterator
  42. #if defined(GENERATING_DOCUMENTATION)
  43. : public std::iterator<
  44. #else // defined(GENERATING_DOCUMENTATION)
  45. : public boost::iterator<
  46. #endif // defined(GENERATING_DOCUMENTATION)
  47. std::forward_iterator_tag,
  48. const basic_resolver_entry<InternetProtocol> >
  49. {
  50. public:
  51. /// Default constructor creates an end iterator.
  52. basic_resolver_iterator()
  53. : index_(0)
  54. {
  55. }
  56. /// Create an iterator from an addrinfo list returned by getaddrinfo.
  57. static basic_resolver_iterator create(
  58. asio::detail::addrinfo_type* address_info,
  59. const std::string& host_name, const std::string& service_name)
  60. {
  61. basic_resolver_iterator iter;
  62. if (!address_info)
  63. return iter;
  64. std::string actual_host_name = host_name;
  65. if (address_info->ai_canonname)
  66. actual_host_name = address_info->ai_canonname;
  67. iter.values_.reset(new values_type);
  68. while (address_info)
  69. {
  70. if (address_info->ai_family == PF_INET
  71. || address_info->ai_family == PF_INET6)
  72. {
  73. using namespace std; // For memcpy.
  74. typename InternetProtocol::endpoint endpoint;
  75. endpoint.resize(static_cast<std::size_t>(address_info->ai_addrlen));
  76. memcpy(endpoint.data(), address_info->ai_addr,
  77. address_info->ai_addrlen);
  78. iter.values_->push_back(
  79. basic_resolver_entry<InternetProtocol>(endpoint,
  80. actual_host_name, service_name));
  81. }
  82. address_info = address_info->ai_next;
  83. }
  84. return iter;
  85. }
  86. /// Create an iterator from an endpoint, host name and service name.
  87. static basic_resolver_iterator create(
  88. const typename InternetProtocol::endpoint& endpoint,
  89. const std::string& host_name, const std::string& service_name)
  90. {
  91. basic_resolver_iterator iter;
  92. iter.values_.reset(new values_type);
  93. iter.values_->push_back(
  94. basic_resolver_entry<InternetProtocol>(
  95. endpoint, host_name, service_name));
  96. return iter;
  97. }
  98. /// Dereference an iterator.
  99. const basic_resolver_entry<InternetProtocol>& operator*() const
  100. {
  101. return dereference();
  102. }
  103. /// Dereference an iterator.
  104. const basic_resolver_entry<InternetProtocol>* operator->() const
  105. {
  106. return &dereference();
  107. }
  108. /// Increment operator (prefix).
  109. basic_resolver_iterator& operator++()
  110. {
  111. increment();
  112. return *this;
  113. }
  114. /// Increment operator (postfix).
  115. basic_resolver_iterator operator++(int)
  116. {
  117. basic_resolver_iterator tmp(*this);
  118. ++*this;
  119. return tmp;
  120. }
  121. /// Test two iterators for equality.
  122. friend bool operator==(const basic_resolver_iterator& a,
  123. const basic_resolver_iterator& b)
  124. {
  125. return a.equal(b);
  126. }
  127. /// Test two iterators for inequality.
  128. friend bool operator!=(const basic_resolver_iterator& a,
  129. const basic_resolver_iterator& b)
  130. {
  131. return !a.equal(b);
  132. }
  133. private:
  134. void increment()
  135. {
  136. if (++index_ == values_->size())
  137. {
  138. // Reset state to match a default constructed end iterator.
  139. values_.reset();
  140. index_ = 0;
  141. }
  142. }
  143. bool equal(const basic_resolver_iterator& other) const
  144. {
  145. if (!values_ && !other.values_)
  146. return true;
  147. if (values_ != other.values_)
  148. return false;
  149. return index_ == other.index_;
  150. }
  151. const basic_resolver_entry<InternetProtocol>& dereference() const
  152. {
  153. return (*values_)[index_];
  154. }
  155. typedef std::vector<basic_resolver_entry<InternetProtocol> > values_type;
  156. boost::shared_ptr<values_type> values_;
  157. std::size_t index_;
  158. };
  159. } // namespace ip
  160. } // namespace asio
  161. #include "asio/detail/pop_options.hpp"
  162. #endif // ASIO_IP_BASIC_RESOLVER_ITERATOR_HPP