mirror of
https://github.com/boostorg/asio.git
synced 2026-01-28 06:42:08 +00:00
........ r43377 | chris_kohlhoff | 2008-02-23 09:43:54 +1100 (Sat, 23 Feb 2008) | 2 lines Use the correct vector of timer queues when dispatching timers. ........ r43437 | chris_kohlhoff | 2008-02-29 23:57:57 +1100 (Fri, 29 Feb 2008) | 2 lines Add missing tie(). ........ r43469 | chris_kohlhoff | 2008-03-04 00:21:05 +1100 (Tue, 04 Mar 2008) | 4 lines Disable use of CancelIo by default, due to the possibility of silent failure on some system configurations. Swallow error returned by CancelIoEx if there are no operations to be cancelled. ........ r43470 | chris_kohlhoff | 2008-03-04 00:27:06 +1100 (Tue, 04 Mar 2008) | 2 lines Add missing 'boost_' prefix to helper namespace. ........ r43471 | chris_kohlhoff | 2008-03-04 00:36:35 +1100 (Tue, 04 Mar 2008) | 2 lines Regenerate documentation. ........ r43472 | chris_kohlhoff | 2008-03-04 01:05:35 +1100 (Tue, 04 Mar 2008) | 1 line Update copyright notices. ........ r43473 | chris_kohlhoff | 2008-03-04 01:13:01 +1100 (Tue, 04 Mar 2008) | 2 lines Update copyright notices. ........ r43569 | chris_kohlhoff | 2008-03-13 00:25:49 +1100 (Thu, 13 Mar 2008) | 4 lines Revert to having the windows-bug workaround (short timeout on GetQueuedCompletionStatus) on all threads as there are still scenarios where threads can get stuck indefinitely. ........ [SVN r43571]
350 lines
11 KiB
C++
350 lines
11 KiB
C++
//
|
|
// multicast.cpp
|
|
// ~~~~~~~~~~~~~
|
|
//
|
|
// Copyright (c) 2003-2008 Christopher M. Kohlhoff (chris at kohlhoff dot com)
|
|
//
|
|
// Distributed under the Boost Software License, Version 1.0. (See accompanying
|
|
// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
|
|
//
|
|
|
|
// Disable autolinking for unit tests.
|
|
#if !defined(BOOST_ALL_NO_LIB)
|
|
#define BOOST_ALL_NO_LIB 1
|
|
#endif // !defined(BOOST_ALL_NO_LIB)
|
|
|
|
// Test that header file is self-contained.
|
|
#include <boost/asio/ip/multicast.hpp>
|
|
|
|
#include <boost/asio.hpp>
|
|
#include "../unit_test.hpp"
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
// ip_multicast_compile test
|
|
// ~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
// The following test checks that all nested classes, enums and constants in
|
|
// ip::multicast compile and link correctly. Runtime failures are ignored.
|
|
|
|
namespace ip_multicast_compile {
|
|
|
|
void test()
|
|
{
|
|
using namespace boost::asio;
|
|
namespace ip = boost::asio::ip;
|
|
|
|
try
|
|
{
|
|
io_service ios;
|
|
ip::udp::socket sock(ios);
|
|
const ip::address address;
|
|
const ip::address_v4 address_v4;
|
|
const ip::address_v6 address_v6;
|
|
|
|
// join_group class.
|
|
|
|
ip::multicast::join_group join_group1;
|
|
ip::multicast::join_group join_group2(address);
|
|
ip::multicast::join_group join_group3(address_v4);
|
|
ip::multicast::join_group join_group4(address_v4, address_v4);
|
|
ip::multicast::join_group join_group5(address_v6);
|
|
ip::multicast::join_group join_group6(address_v6, 1);
|
|
sock.set_option(join_group6);
|
|
|
|
// leave_group class.
|
|
|
|
ip::multicast::leave_group leave_group1;
|
|
ip::multicast::leave_group leave_group2(address);
|
|
ip::multicast::leave_group leave_group3(address_v4);
|
|
ip::multicast::leave_group leave_group4(address_v4, address_v4);
|
|
ip::multicast::leave_group leave_group5(address_v6);
|
|
ip::multicast::leave_group leave_group6(address_v6, 1);
|
|
sock.set_option(leave_group6);
|
|
|
|
// outbound_interface class.
|
|
|
|
ip::multicast::outbound_interface outbound_interface1;
|
|
ip::multicast::outbound_interface outbound_interface2(address_v4);
|
|
ip::multicast::outbound_interface outbound_interface3(1);
|
|
sock.set_option(outbound_interface3);
|
|
|
|
// hops class.
|
|
|
|
ip::multicast::hops hops1(1024);
|
|
sock.set_option(hops1);
|
|
ip::multicast::hops hops2;
|
|
sock.get_option(hops2);
|
|
hops1 = 1;
|
|
static_cast<int>(hops1.value());
|
|
|
|
// enable_loopback class.
|
|
|
|
ip::multicast::enable_loopback enable_loopback1(true);
|
|
sock.set_option(enable_loopback1);
|
|
ip::multicast::enable_loopback enable_loopback2;
|
|
sock.get_option(enable_loopback2);
|
|
enable_loopback1 = true;
|
|
static_cast<bool>(enable_loopback1);
|
|
static_cast<bool>(!enable_loopback1);
|
|
static_cast<bool>(enable_loopback1.value());
|
|
}
|
|
catch (std::exception&)
|
|
{
|
|
}
|
|
}
|
|
|
|
} // namespace ip_multicast_compile
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
// ip_multicast_runtime test
|
|
// ~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
// The following test checks the runtime operation of the socket options defined
|
|
// in the ip::multicast namespace.
|
|
|
|
namespace ip_multicast_runtime {
|
|
|
|
#if defined(__hpux)
|
|
// HP-UX doesn't declare this function extern "C", so it is declared again here
|
|
// to avoid a linker error about an undefined symbol.
|
|
extern "C" unsigned int if_nametoindex(const char*);
|
|
#endif // defined(__hpux)
|
|
|
|
void test()
|
|
{
|
|
using namespace boost::asio;
|
|
namespace ip = boost::asio::ip;
|
|
|
|
io_service ios;
|
|
boost::system::error_code ec;
|
|
|
|
ip::udp::endpoint ep_v4(ip::address_v4::loopback(), 0);
|
|
ip::udp::socket sock_v4(ios);
|
|
sock_v4.open(ep_v4.protocol(), ec);
|
|
sock_v4.bind(ep_v4, ec);
|
|
bool have_v4 = !ec;
|
|
|
|
ip::udp::endpoint ep_v6(ip::address_v6::loopback(), 0);
|
|
ip::udp::socket sock_v6(ios);
|
|
sock_v6.open(ep_v6.protocol(), ec);
|
|
sock_v6.bind(ep_v6, ec);
|
|
bool have_v6 = !ec;
|
|
|
|
BOOST_CHECK(have_v4 || have_v6);
|
|
|
|
#if defined(BOOST_WINDOWS) && defined(UNDER_CE)
|
|
// Windows CE seems to have problems with some multicast group addresses.
|
|
// The following address works on CE, but as it is not a private multicast
|
|
// address it will not be used on other platforms.
|
|
const ip::address multicast_address_v4 =
|
|
ip::address::from_string("239.0.0.4", ec);
|
|
#else // defined(BOOST_WINDOWS) && defined(UNDER_CE)
|
|
const ip::address multicast_address_v4 =
|
|
ip::address::from_string("239.255.0.1", ec);
|
|
#endif // defined(BOOST_WINDOWS) && defined(UNDER_CE)
|
|
BOOST_CHECK(!have_v4 || !ec);
|
|
|
|
const ip::address multicast_address_v6 =
|
|
ip::address::from_string("ff01::1", ec);
|
|
BOOST_CHECK(!have_v6 || !ec);
|
|
|
|
// join_group class.
|
|
|
|
if (have_v4)
|
|
{
|
|
ip::multicast::join_group join_group(multicast_address_v4);
|
|
sock_v4.set_option(join_group, ec);
|
|
BOOST_CHECK_MESSAGE(!ec, ec.value() << ", " << ec.message());
|
|
}
|
|
|
|
if (have_v6)
|
|
{
|
|
ip::multicast::join_group join_group(multicast_address_v6);
|
|
sock_v6.set_option(join_group, ec);
|
|
BOOST_CHECK_MESSAGE(!ec, ec.value() << ", " << ec.message());
|
|
}
|
|
|
|
// leave_group class.
|
|
|
|
if (have_v4)
|
|
{
|
|
ip::multicast::leave_group leave_group(multicast_address_v4);
|
|
sock_v4.set_option(leave_group, ec);
|
|
BOOST_CHECK_MESSAGE(!ec, ec.value() << ", " << ec.message());
|
|
}
|
|
|
|
if (have_v6)
|
|
{
|
|
ip::multicast::leave_group leave_group(multicast_address_v6);
|
|
sock_v6.set_option(leave_group, ec);
|
|
BOOST_CHECK_MESSAGE(!ec, ec.value() << ", " << ec.message());
|
|
}
|
|
|
|
// outbound_interface class.
|
|
|
|
if (have_v4)
|
|
{
|
|
ip::multicast::outbound_interface outbound_interface(
|
|
ip::address_v4::loopback());
|
|
sock_v4.set_option(outbound_interface, ec);
|
|
BOOST_CHECK_MESSAGE(!ec, ec.value() << ", " << ec.message());
|
|
}
|
|
|
|
if (have_v6)
|
|
{
|
|
#if defined(__hpux)
|
|
ip::multicast::outbound_interface outbound_interface(if_nametoindex("lo0"));
|
|
#else
|
|
ip::multicast::outbound_interface outbound_interface(1);
|
|
#endif
|
|
sock_v6.set_option(outbound_interface, ec);
|
|
BOOST_CHECK_MESSAGE(!ec, ec.value() << ", " << ec.message());
|
|
}
|
|
|
|
// hops class.
|
|
|
|
if (have_v4)
|
|
{
|
|
ip::multicast::hops hops1(1);
|
|
BOOST_CHECK(hops1.value() == 1);
|
|
sock_v4.set_option(hops1, ec);
|
|
BOOST_CHECK_MESSAGE(!ec, ec.value() << ", " << ec.message());
|
|
|
|
ip::multicast::hops hops2;
|
|
sock_v4.get_option(hops2, ec);
|
|
BOOST_CHECK_MESSAGE(!ec, ec.value() << ", " << ec.message());
|
|
BOOST_CHECK(hops2.value() == 1);
|
|
|
|
ip::multicast::hops hops3(0);
|
|
BOOST_CHECK(hops3.value() == 0);
|
|
sock_v4.set_option(hops3, ec);
|
|
BOOST_CHECK_MESSAGE(!ec, ec.value() << ", " << ec.message());
|
|
|
|
ip::multicast::hops hops4;
|
|
sock_v4.get_option(hops4, ec);
|
|
BOOST_CHECK_MESSAGE(!ec, ec.value() << ", " << ec.message());
|
|
BOOST_CHECK(hops4.value() == 0);
|
|
}
|
|
|
|
if (have_v6)
|
|
{
|
|
ip::multicast::hops hops1(1);
|
|
BOOST_CHECK(hops1.value() == 1);
|
|
sock_v6.set_option(hops1, ec);
|
|
BOOST_CHECK_MESSAGE(!ec, ec.value() << ", " << ec.message());
|
|
|
|
ip::multicast::hops hops2;
|
|
sock_v6.get_option(hops2, ec);
|
|
BOOST_CHECK_MESSAGE(!ec, ec.value() << ", " << ec.message());
|
|
BOOST_CHECK(hops2.value() == 1);
|
|
|
|
ip::multicast::hops hops3(0);
|
|
BOOST_CHECK(hops3.value() == 0);
|
|
sock_v6.set_option(hops3, ec);
|
|
BOOST_CHECK_MESSAGE(!ec, ec.value() << ", " << ec.message());
|
|
|
|
ip::multicast::hops hops4;
|
|
sock_v6.get_option(hops4, ec);
|
|
BOOST_CHECK_MESSAGE(!ec, ec.value() << ", " << ec.message());
|
|
BOOST_CHECK(hops4.value() == 0);
|
|
}
|
|
|
|
// enable_loopback class.
|
|
|
|
if (have_v4)
|
|
{
|
|
ip::multicast::enable_loopback enable_loopback1(true);
|
|
BOOST_CHECK(enable_loopback1.value());
|
|
BOOST_CHECK(static_cast<bool>(enable_loopback1));
|
|
BOOST_CHECK(!!enable_loopback1);
|
|
sock_v4.set_option(enable_loopback1, ec);
|
|
#if defined(BOOST_WINDOWS) && defined(UNDER_CE)
|
|
// Option is not supported under Windows CE.
|
|
BOOST_CHECK_MESSAGE(ec == boost::asio::error::no_protocol_option,
|
|
ec.value() << ", " << ec.message());
|
|
#else // defined(BOOST_WINDOWS) && defined(UNDER_CE)
|
|
BOOST_CHECK_MESSAGE(!ec, ec.value() << ", " << ec.message());
|
|
#endif // defined(BOOST_WINDOWS) && defined(UNDER_CE)
|
|
|
|
ip::multicast::enable_loopback enable_loopback2;
|
|
sock_v4.get_option(enable_loopback2, ec);
|
|
#if defined(BOOST_WINDOWS) && defined(UNDER_CE)
|
|
// Not supported under Windows CE but can get value.
|
|
BOOST_CHECK_MESSAGE(!ec, ec.value() << ", " << ec.message());
|
|
#else // defined(BOOST_WINDOWS) && defined(UNDER_CE)
|
|
BOOST_CHECK_MESSAGE(!ec, ec.value() << ", " << ec.message());
|
|
BOOST_CHECK(enable_loopback2.value());
|
|
BOOST_CHECK(static_cast<bool>(enable_loopback2));
|
|
BOOST_CHECK(!!enable_loopback2);
|
|
#endif // defined(BOOST_WINDOWS) && defined(UNDER_CE)
|
|
|
|
ip::multicast::enable_loopback enable_loopback3(false);
|
|
BOOST_CHECK(!enable_loopback3.value());
|
|
BOOST_CHECK(!static_cast<bool>(enable_loopback3));
|
|
BOOST_CHECK(!enable_loopback3);
|
|
sock_v4.set_option(enable_loopback3, ec);
|
|
#if defined(BOOST_WINDOWS) && defined(UNDER_CE)
|
|
// Option is not supported under Windows CE.
|
|
BOOST_CHECK_MESSAGE(ec == boost::asio::error::no_protocol_option,
|
|
ec.value() << ", " << ec.message());
|
|
#else // defined(BOOST_WINDOWS) && defined(UNDER_CE)
|
|
BOOST_CHECK_MESSAGE(!ec, ec.value() << ", " << ec.message());
|
|
#endif // defined(BOOST_WINDOWS) && defined(UNDER_CE)
|
|
|
|
ip::multicast::enable_loopback enable_loopback4;
|
|
sock_v4.get_option(enable_loopback4, ec);
|
|
#if defined(BOOST_WINDOWS) && defined(UNDER_CE)
|
|
// Not supported under Windows CE but can get value.
|
|
BOOST_CHECK_MESSAGE(!ec, ec.value() << ", " << ec.message());
|
|
#else // defined(BOOST_WINDOWS) && defined(UNDER_CE)
|
|
BOOST_CHECK_MESSAGE(!ec, ec.value() << ", " << ec.message());
|
|
BOOST_CHECK(!enable_loopback4.value());
|
|
BOOST_CHECK(!static_cast<bool>(enable_loopback4));
|
|
BOOST_CHECK(!enable_loopback4);
|
|
#endif // defined(BOOST_WINDOWS) && defined(UNDER_CE)
|
|
}
|
|
|
|
if (have_v6)
|
|
{
|
|
ip::multicast::enable_loopback enable_loopback1(true);
|
|
BOOST_CHECK(enable_loopback1.value());
|
|
BOOST_CHECK(static_cast<bool>(enable_loopback1));
|
|
BOOST_CHECK(!!enable_loopback1);
|
|
sock_v6.set_option(enable_loopback1, ec);
|
|
BOOST_CHECK_MESSAGE(!ec, ec.value() << ", " << ec.message());
|
|
|
|
ip::multicast::enable_loopback enable_loopback2;
|
|
sock_v6.get_option(enable_loopback2, ec);
|
|
BOOST_CHECK_MESSAGE(!ec, ec.value() << ", " << ec.message());
|
|
BOOST_CHECK(enable_loopback2.value());
|
|
BOOST_CHECK(static_cast<bool>(enable_loopback2));
|
|
BOOST_CHECK(!!enable_loopback2);
|
|
|
|
ip::multicast::enable_loopback enable_loopback3(false);
|
|
BOOST_CHECK(!enable_loopback3.value());
|
|
BOOST_CHECK(!static_cast<bool>(enable_loopback3));
|
|
BOOST_CHECK(!enable_loopback3);
|
|
sock_v6.set_option(enable_loopback3, ec);
|
|
BOOST_CHECK_MESSAGE(!ec, ec.value() << ", " << ec.message());
|
|
|
|
ip::multicast::enable_loopback enable_loopback4;
|
|
sock_v6.get_option(enable_loopback4, ec);
|
|
BOOST_CHECK_MESSAGE(!ec, ec.value() << ", " << ec.message());
|
|
BOOST_CHECK(!enable_loopback4.value());
|
|
BOOST_CHECK(!static_cast<bool>(enable_loopback4));
|
|
BOOST_CHECK(!enable_loopback4);
|
|
}
|
|
}
|
|
|
|
} // namespace ip_multicast_runtime
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
test_suite* init_unit_test_suite(int argc, char* argv[])
|
|
{
|
|
test_suite* test = BOOST_TEST_SUITE("ip/multicast");
|
|
test->add(BOOST_TEST_CASE(&ip_multicast_compile::test));
|
|
test->add(BOOST_TEST_CASE(&ip_multicast_runtime::test));
|
|
return test;
|
|
}
|