|  | // Copyright 2013 The Chromium Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | #include "net/dns/mdns_client.h" | 
|  |  | 
|  | #include "net/base/net_errors.h" | 
|  | #include "net/base/net_util.h" | 
|  | #include "net/base/network_interfaces.h" | 
|  | #include "net/dns/dns_protocol.h" | 
|  | #include "net/dns/mdns_client_impl.h" | 
|  |  | 
|  | namespace net { | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | const char kMDnsMulticastGroupIPv4[] = "224.0.0.251"; | 
|  | const char kMDnsMulticastGroupIPv6[] = "FF02::FB"; | 
|  |  | 
|  | IPEndPoint GetMDnsIPEndPoint(const char* address) { | 
|  | IPAddressNumber multicast_group_number; | 
|  | bool success = ParseIPLiteralToNumber(address, | 
|  | &multicast_group_number); | 
|  | DCHECK(success); | 
|  | return IPEndPoint(multicast_group_number, | 
|  | dns_protocol::kDefaultPortMulticast); | 
|  | } | 
|  |  | 
|  | int Bind(const IPEndPoint& multicast_addr, | 
|  | uint32_t interface_index, | 
|  | DatagramServerSocket* socket) { | 
|  | IPAddressNumber address_any(multicast_addr.address().size()); | 
|  | IPEndPoint bind_endpoint(address_any, multicast_addr.port()); | 
|  |  | 
|  | socket->AllowAddressReuse(); | 
|  | socket->SetMulticastInterface(interface_index); | 
|  |  | 
|  | int rv = socket->Listen(bind_endpoint); | 
|  | if (rv < OK) | 
|  | return rv; | 
|  |  | 
|  | return socket->JoinGroup(multicast_addr.address().bytes()); | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | // static | 
|  | scoped_ptr<MDnsSocketFactory> MDnsSocketFactory::CreateDefault() { | 
|  | return scoped_ptr<MDnsSocketFactory>(new MDnsSocketFactoryImpl); | 
|  | } | 
|  |  | 
|  | // static | 
|  | scoped_ptr<MDnsClient> MDnsClient::CreateDefault() { | 
|  | return scoped_ptr<MDnsClient>(new MDnsClientImpl()); | 
|  | } | 
|  |  | 
|  | IPEndPoint GetMDnsIPEndPoint(AddressFamily address_family) { | 
|  | switch (address_family) { | 
|  | case ADDRESS_FAMILY_IPV4: | 
|  | return GetMDnsIPEndPoint(kMDnsMulticastGroupIPv4); | 
|  | case ADDRESS_FAMILY_IPV6: | 
|  | return GetMDnsIPEndPoint(kMDnsMulticastGroupIPv6); | 
|  | default: | 
|  | NOTREACHED(); | 
|  | return IPEndPoint(); | 
|  | } | 
|  | } | 
|  |  | 
|  | InterfaceIndexFamilyList GetMDnsInterfacesToBind() { | 
|  | NetworkInterfaceList network_list; | 
|  | InterfaceIndexFamilyList interfaces; | 
|  | if (!GetNetworkList(&network_list, INCLUDE_HOST_SCOPE_VIRTUAL_INTERFACES)) | 
|  | return interfaces; | 
|  | for (size_t i = 0; i < network_list.size(); ++i) { | 
|  | AddressFamily family = GetAddressFamily(network_list[i].address); | 
|  | if (family == ADDRESS_FAMILY_IPV4 || family == ADDRESS_FAMILY_IPV6) { | 
|  | interfaces.push_back( | 
|  | std::make_pair(network_list[i].interface_index, family)); | 
|  | } | 
|  | } | 
|  | std::sort(interfaces.begin(), interfaces.end()); | 
|  | // Interfaces could have multiple addresses. Filter out duplicate entries. | 
|  | interfaces.erase(std::unique(interfaces.begin(), interfaces.end()), | 
|  | interfaces.end()); | 
|  | return interfaces; | 
|  | } | 
|  |  | 
|  | scoped_ptr<DatagramServerSocket> CreateAndBindMDnsSocket( | 
|  | AddressFamily address_family, | 
|  | uint32_t interface_index) { | 
|  | scoped_ptr<DatagramServerSocket> socket( | 
|  | new UDPServerSocket(NULL, NetLog::Source())); | 
|  |  | 
|  | IPEndPoint multicast_addr = GetMDnsIPEndPoint(address_family); | 
|  | int rv = Bind(multicast_addr, interface_index, socket.get()); | 
|  | if (rv != OK) { | 
|  | socket.reset(); | 
|  | VLOG(1) << "Bind failed, endpoint=" << multicast_addr.ToStringWithoutPort() | 
|  | << ", error=" << rv; | 
|  | } | 
|  | return socket; | 
|  | } | 
|  |  | 
|  | }  // namespace net |