/* * Copyright 2017, 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 "interface.h" #include <errno.h> #include <linux/if.h> #include <linux/if_ether.h> #include <linux/route.h> #include <string.h> #include <unistd.h> Interface::Interface() : mSocketFd(-1) { } Interface::~Interface() { if (mSocketFd != -1) { close(mSocketFd); mSocketFd = -1; } } Result Interface::init(const char* interfaceName) { mInterfaceName = interfaceName; if (mSocketFd != -1) { return Result::error("Interface initialized more than once"); } mSocketFd = ::socket(AF_INET, SOCK_DGRAM | SOCK_CLOEXEC, IPPROTO_IP); if (mSocketFd == -1) { return Result::error("Failed to create interface socket for '%s': %s", interfaceName, strerror(errno)); } Result res = populateIndex(); if (!res) { return res; } res = populateMacAddress(); if (!res) { return res; } res = bringUp(); if (!res) { return res; } res = setAddress(0); if (!res) { return res; } return Result::success(); } Result Interface::bringUp() { return setInterfaceUp(true); } Result Interface::bringDown() { return setInterfaceUp(false); } Result Interface::setMtu(uint16_t mtu) { struct ifreq request = createRequest(); strncpy(request.ifr_name, mInterfaceName.c_str(), sizeof(request.ifr_name)); request.ifr_mtu = mtu; int status = ::ioctl(mSocketFd, SIOCSIFMTU, &request); if (status != 0) { return Result::error("Failed to set interface MTU %u for '%s': %s", static_cast<unsigned int>(mtu), mInterfaceName.c_str(), strerror(errno)); } return Result::success(); } Result Interface::setAddress(in_addr_t address) { struct ifreq request = createRequest(); auto requestAddr = reinterpret_cast<struct sockaddr_in*>(&request.ifr_addr); requestAddr->sin_family = AF_INET; requestAddr->sin_port = 0; requestAddr->sin_addr.s_addr = address; int status = ::ioctl(mSocketFd, SIOCSIFADDR, &request); if (status != 0) { return Result::error("Failed to set interface address for '%s': %s", mInterfaceName.c_str(), strerror(errno)); } return Result::success(); } Result Interface::setSubnetMask(in_addr_t subnetMask) { struct ifreq request = createRequest(); auto addr = reinterpret_cast<struct sockaddr_in*>(&request.ifr_addr); addr->sin_family = AF_INET; addr->sin_port = 0; addr->sin_addr.s_addr = subnetMask; int status = ::ioctl(mSocketFd, SIOCSIFNETMASK, &request); if (status != 0) { return Result::error("Failed to set subnet mask for '%s': %s", mInterfaceName.c_str(), strerror(errno)); } return Result::success(); } struct ifreq Interface::createRequest() const { struct ifreq request; memset(&request, 0, sizeof(request)); strncpy(request.ifr_name, mInterfaceName.c_str(), sizeof(request.ifr_name)); request.ifr_name[sizeof(request.ifr_name) - 1] = '\0'; return request; } Result Interface::populateIndex() { struct ifreq request = createRequest(); int status = ::ioctl(mSocketFd, SIOCGIFINDEX, &request); if (status != 0) { return Result::error("Failed to get interface index for '%s': %s", mInterfaceName.c_str(), strerror(errno)); } mIndex = request.ifr_ifindex; return Result::success(); } Result Interface::populateMacAddress() { struct ifreq request = createRequest(); int status = ::ioctl(mSocketFd, SIOCGIFHWADDR, &request); if (status != 0) { return Result::error("Failed to get MAC address for '%s': %s", mInterfaceName.c_str(), strerror(errno)); } memcpy(mMacAddress, &request.ifr_hwaddr.sa_data, ETH_ALEN); return Result::success(); } Result Interface::setInterfaceUp(bool shouldBeUp) { struct ifreq request = createRequest(); int status = ::ioctl(mSocketFd, SIOCGIFFLAGS, &request); if (status != 0) { return Result::error("Failed to get interface flags for '%s': %s", mInterfaceName.c_str(), strerror(errno)); } bool isUp = (request.ifr_flags & IFF_UP) != 0; if (isUp != shouldBeUp) { // Toggle the up flag request.ifr_flags ^= IFF_UP; } else { // Interface is already in desired state, do nothing return Result::success(); } status = ::ioctl(mSocketFd, SIOCSIFFLAGS, &request); if (status != 0) { return Result::error("Failed to set interface flags for '%s': %s", mInterfaceName.c_str(), strerror(errno)); } return Result::success(); }