basic_resolver_iterator.hpp 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156
  1. //
  2. // basic_resolver_iterator.hpp
  3. // ~~~~~~~~~~~~~~~~~~~~~~~~~~~
  4. //
  5. // Copyright (c) 2003-2008 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 BOOST_ASIO_IP_BASIC_RESOLVER_ITERATOR_HPP
  11. #define BOOST_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 <boost/asio/detail/push_options.hpp>
  16. #include <boost/asio/detail/push_options.hpp>
  17. #include <boost/iterator/iterator_facade.hpp>
  18. #include <boost/optional.hpp>
  19. #include <boost/shared_ptr.hpp>
  20. #include <cstring>
  21. #include <string>
  22. #include <vector>
  23. #include <boost/asio/detail/pop_options.hpp>
  24. #include <boost/asio/detail/socket_ops.hpp>
  25. #include <boost/asio/detail/socket_types.hpp>
  26. #include <boost/asio/ip/basic_resolver_entry.hpp>
  27. namespace boost {
  28. namespace asio {
  29. namespace ip {
  30. /// An iterator over the entries produced by a resolver.
  31. /**
  32. * The boost::asio::ip::basic_resolver_iterator class template is used to define
  33. * iterators over the results returned by a resolver.
  34. *
  35. * The iterator's value_type, obtained when the iterator is dereferenced, is:
  36. * @code const basic_resolver_entry<InternetProtocol> @endcode
  37. *
  38. * @par Thread Safety
  39. * @e Distinct @e objects: Safe.@n
  40. * @e Shared @e objects: Unsafe.
  41. */
  42. template <typename InternetProtocol>
  43. class basic_resolver_iterator
  44. : public boost::iterator_facade<
  45. basic_resolver_iterator<InternetProtocol>,
  46. const basic_resolver_entry<InternetProtocol>,
  47. boost::forward_traversal_tag>
  48. {
  49. public:
  50. /// Default constructor creates an end iterator.
  51. basic_resolver_iterator()
  52. {
  53. }
  54. /// Create an iterator from an addrinfo list returned by getaddrinfo.
  55. static basic_resolver_iterator create(
  56. boost::asio::detail::addrinfo_type* address_info,
  57. const std::string& host_name, const std::string& service_name)
  58. {
  59. basic_resolver_iterator iter;
  60. if (!address_info)
  61. return iter;
  62. std::string actual_host_name = host_name;
  63. if (address_info->ai_canonname)
  64. actual_host_name = address_info->ai_canonname;
  65. iter.values_.reset(new values_type);
  66. while (address_info)
  67. {
  68. if (address_info->ai_family == PF_INET
  69. || address_info->ai_family == PF_INET6)
  70. {
  71. using namespace std; // For memcpy.
  72. typename InternetProtocol::endpoint endpoint;
  73. endpoint.resize(static_cast<std::size_t>(address_info->ai_addrlen));
  74. memcpy(endpoint.data(), address_info->ai_addr,
  75. address_info->ai_addrlen);
  76. iter.values_->push_back(
  77. basic_resolver_entry<InternetProtocol>(endpoint,
  78. actual_host_name, service_name));
  79. }
  80. address_info = address_info->ai_next;
  81. }
  82. if (iter.values_->size())
  83. iter.iter_ = iter.values_->begin();
  84. else
  85. iter.values_.reset();
  86. return iter;
  87. }
  88. /// Create an iterator from an endpoint, host name and service name.
  89. static basic_resolver_iterator create(
  90. const typename InternetProtocol::endpoint& endpoint,
  91. const std::string& host_name, const std::string& service_name)
  92. {
  93. basic_resolver_iterator iter;
  94. iter.values_.reset(new values_type);
  95. iter.values_->push_back(
  96. basic_resolver_entry<InternetProtocol>(
  97. endpoint, host_name, service_name));
  98. iter.iter_ = iter.values_->begin();
  99. return iter;
  100. }
  101. private:
  102. friend class boost::iterator_core_access;
  103. void increment()
  104. {
  105. if (++*iter_ == values_->end())
  106. {
  107. // Reset state to match a default constructed end iterator.
  108. values_.reset();
  109. typedef typename values_type::const_iterator values_iterator_type;
  110. iter_.reset();
  111. }
  112. }
  113. bool equal(const basic_resolver_iterator& other) const
  114. {
  115. if (!values_ && !other.values_)
  116. return true;
  117. if (values_ != other.values_)
  118. return false;
  119. return *iter_ == *other.iter_;
  120. }
  121. const basic_resolver_entry<InternetProtocol>& dereference() const
  122. {
  123. return **iter_;
  124. }
  125. typedef std::vector<basic_resolver_entry<InternetProtocol> > values_type;
  126. typedef typename values_type::const_iterator values_iter_type;
  127. boost::shared_ptr<values_type> values_;
  128. boost::optional<values_iter_type> iter_;
  129. };
  130. } // namespace ip
  131. } // namespace asio
  132. } // namespace boost
  133. #include <boost/asio/detail/pop_options.hpp>
  134. #endif // BOOST_ASIO_IP_BASIC_RESOLVER_ITERATOR_HPP