// // Copyright (C) 2015 The Android Open Source Project // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. // #include "dhcp_client/device_info.h" #include <net/if.h> #include <netinet/in.h> #include <sys/ioctl.h> #include <memory> #include <string> #include <base/logging.h> using shill::ByteString; using shill::Sockets; using shill::RTNLHandler; using std::unique_ptr; namespace { base::LazyInstance<dhcp_client::DeviceInfo> g_dhcp_device_info = LAZY_INSTANCE_INITIALIZER; } // namespace namespace dhcp_client { DeviceInfo::DeviceInfo() : sockets_(new Sockets()), rtnl_handler_(RTNLHandler::GetInstance()) { } DeviceInfo::~DeviceInfo() {} DeviceInfo* DeviceInfo::GetInstance() { return g_dhcp_device_info.Pointer(); } bool DeviceInfo::GetDeviceInfo(const std::string& interface_name, ByteString* mac_address, unsigned int* interface_index ) { struct ifreq ifr; size_t if_name_len = interface_name.size(); if (if_name_len > IFNAMSIZ) { LOG(ERROR) << "Interface name is too long."; return false; } memcpy(ifr.ifr_name, interface_name.c_str(), if_name_len); ifr.ifr_name[if_name_len] = 0; int fd = sockets_->Socket(AF_INET, SOCK_DGRAM, 0); if (fd == -1) { PLOG(ERROR) << "Failed to create socket."; return false; } shill::ScopedSocketCloser socket_closer(sockets_.get(), fd); // Get interface hardware address if (sockets_->Ioctl(fd, SIOCGIFHWADDR, &ifr) == -1) { PLOG(ERROR) << "Failed to get interface hardware address."; return false; } int if_index = rtnl_handler_->GetInterfaceIndex(interface_name); if (if_index == -1) { LOG(ERROR) << "Unable to get interface index."; return false; } *interface_index = if_index; *mac_address = ByteString(ifr.ifr_hwaddr.sa_data, IFHWADDRLEN); return true; } } // namespace dhcp_client