This source file includes following definitions.
- RemovePermanentIPv6AddressesWhereTemporaryExists
- FileURLToFilePath
- GetNetworkList
- GetWifiPHYLayerProtocol
#include "net/base/net_util.h"
#include <set>
#include <sys/types.h>
#include "base/files/file_path.h"
#include "base/logging.h"
#include "base/strings/string_number_conversions.h"
#include "base/strings/string_tokenizer.h"
#include "base/strings/string_util.h"
#include "base/threading/thread_restrictions.h"
#include "net/base/escape.h"
#include "net/base/ip_endpoint.h"
#include "net/base/net_errors.h"
#include "url/gurl.h"
#if !defined(OS_ANDROID)
#include <ifaddrs.h>
#endif
#include <net/if.h>
#include <netinet/in.h>
#if defined(OS_MACOSX) && !defined(OS_IOS)
#include <netinet/in_var.h>
#include <sys/ioctl.h>
#endif
#if defined(OS_ANDROID)
#include "net/android/network_library.h"
#endif
namespace net {
namespace {
#if !defined(OS_ANDROID)
struct NetworkInterfaceInfo {
NetworkInterfaceInfo() : permanent(true) { }
bool permanent;
NetworkInterface interface;
};
void RemovePermanentIPv6AddressesWhereTemporaryExists(
std::vector<NetworkInterfaceInfo>* infos) {
if (!infos || infos->empty())
return;
std::set<std::string> ifaces_with_temp_addrs;
std::vector<NetworkInterfaceInfo>::iterator i;
for (i = infos->begin(); i != infos->end(); ++i) {
if (!i->permanent && i->interface.address.size() == kIPv6AddressSize) {
ifaces_with_temp_addrs.insert(i->interface.name);
}
}
if (ifaces_with_temp_addrs.empty())
return;
for (i = infos->begin(); i != infos->end(); ) {
if ((i->interface.address.size() == kIPv6AddressSize) && i->permanent &&
(ifaces_with_temp_addrs.find(i->interface.name) !=
ifaces_with_temp_addrs.end())) {
i = infos->erase(i);
} else {
++i;
}
}
}
#endif
}
bool FileURLToFilePath(const GURL& url, base::FilePath* path) {
*path = base::FilePath();
std::string& file_path_str = const_cast<std::string&>(path->value());
file_path_str.clear();
if (!url.is_valid())
return false;
std::string old_path = url.path();
if (old_path.empty())
return false;
old_path = UnescapeURLComponent(old_path,
UnescapeRule::SPACES | UnescapeRule::URL_SPECIAL_CHARS);
std::string new_path;
do {
new_path = old_path;
ReplaceSubstringsAfterOffset(&new_path, 0, "//", "/");
old_path.swap(new_path);
} while (new_path != old_path);
file_path_str.assign(old_path);
return !file_path_str.empty();
}
bool GetNetworkList(NetworkInterfaceList* networks, int policy) {
#if defined(OS_ANDROID)
std::string network_list = android::GetNetworkList();
base::StringTokenizer network_interfaces(network_list, "\n");
while (network_interfaces.GetNext()) {
std::string network_item = network_interfaces.token();
base::StringTokenizer network_tokenizer(network_item, "\t");
CHECK(network_tokenizer.GetNext());
std::string name = network_tokenizer.token();
CHECK(network_tokenizer.GetNext());
std::string interface_address = network_tokenizer.token();
IPAddressNumber address;
size_t network_prefix = 0;
CHECK(ParseCIDRBlock(network_tokenizer.token(),
&address,
&network_prefix));
CHECK(network_tokenizer.GetNext());
uint32 index = 0;
CHECK(base::StringToUint(network_tokenizer.token(), &index));
networks->push_back(
NetworkInterface(name, name, index, address, network_prefix));
}
return true;
#else
base::ThreadRestrictions::AssertIOAllowed();
int ioctl_socket = -1;
if (policy & INCLUDE_ONLY_TEMP_IPV6_ADDRESS_IF_POSSIBLE) {
ioctl_socket = socket(AF_INET6, SOCK_DGRAM, 0);
DCHECK_GT(ioctl_socket, 0);
}
ifaddrs *interfaces;
if (getifaddrs(&interfaces) < 0) {
PLOG(ERROR) << "getifaddrs";
return false;
}
std::vector<NetworkInterfaceInfo> network_infos;
for (ifaddrs *interface = interfaces;
interface != NULL;
interface = interface->ifa_next) {
if (!(IFF_UP & interface->ifa_flags))
continue;
if (IFF_LOOPBACK & interface->ifa_flags)
continue;
struct sockaddr* addr = interface->ifa_addr;
if (!addr)
continue;
int addr_size = 0;
if (addr->sa_family == AF_INET6) {
struct sockaddr_in6* addr_in6 =
reinterpret_cast<struct sockaddr_in6*>(addr);
struct in6_addr* sin6_addr = &addr_in6->sin6_addr;
addr_size = sizeof(*addr_in6);
if (IN6_IS_ADDR_LOOPBACK(sin6_addr) ||
IN6_IS_ADDR_UNSPECIFIED(sin6_addr)) {
continue;
}
} else if (addr->sa_family == AF_INET) {
struct sockaddr_in* addr_in =
reinterpret_cast<struct sockaddr_in*>(addr);
addr_size = sizeof(*addr_in);
if (addr_in->sin_addr.s_addr == INADDR_LOOPBACK ||
addr_in->sin_addr.s_addr == 0) {
continue;
}
} else {
continue;
}
const std::string& name = interface->ifa_name;
if ((policy & EXCLUDE_HOST_SCOPE_VIRTUAL_INTERFACES) &&
((name.find("vmnet") != std::string::npos) ||
(name.find("vnic") != std::string::npos))) {
continue;
}
NetworkInterfaceInfo network_info;
#if defined(OS_MACOSX) && !defined(OS_IOS)
if ((policy & INCLUDE_ONLY_TEMP_IPV6_ADDRESS_IF_POSSIBLE) &&
ioctl_socket >= 0 && addr->sa_family == AF_INET6) {
struct in6_ifreq ifr = {};
strncpy(ifr.ifr_name, interface->ifa_name, sizeof(ifr.ifr_name) - 1);
memcpy(&ifr.ifr_ifru.ifru_addr, interface->ifa_addr,
interface->ifa_addr->sa_len);
int rv = ioctl(ioctl_socket, SIOCGIFAFLAG_IN6, &ifr);
if (rv >= 0) {
network_info.permanent = !(ifr.ifr_ifru.ifru_flags & IN6_IFF_TEMPORARY);
}
}
#endif
IPEndPoint address;
if (address.FromSockAddr(addr, addr_size)) {
uint8 net_mask = 0;
if (interface->ifa_netmask) {
if (interface->ifa_netmask->sa_family == 0) {
interface->ifa_netmask->sa_family = addr->sa_family;
}
IPEndPoint netmask;
if (netmask.FromSockAddr(interface->ifa_netmask, addr_size)) {
net_mask = MaskPrefixLength(netmask.address());
}
}
network_info.interface = NetworkInterface(
name, name, if_nametoindex(name.c_str()),
address.address(), net_mask);
network_infos.push_back(NetworkInterfaceInfo(network_info));
}
}
freeifaddrs(interfaces);
if (ioctl_socket >= 0) {
close(ioctl_socket);
}
if (policy & INCLUDE_ONLY_TEMP_IPV6_ADDRESS_IF_POSSIBLE) {
RemovePermanentIPv6AddressesWhereTemporaryExists(&network_infos);
}
for (size_t i = 0; i < network_infos.size(); ++i) {
networks->push_back(network_infos[i].interface);
}
return true;
#endif
}
WifiPHYLayerProtocol GetWifiPHYLayerProtocol() {
return WIFI_PHY_LAYER_PROTOCOL_UNKNOWN;
}
}