/*
* 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();
}