/* vim: set expandtab ts=4 sw=4: */
/*
* You may redistribute this program and/or modify it under the terms of
* the GNU General Public License as published by the Free Software Foundation,
* either version 3 of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
#include "exception/Except.h"
#include "memory/Allocator.h"
#include "util/events/EventBase.h"
#include "interface/Interface.h"
#include "interface/tuntap/TUNInterface.h"
#include "util/events/Pipe.h"
#define string_strncpy
#define string_strlen
#define string_strerror
#include "util/platform/libc/string.h"
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
struct Interface* TUNInterface_new(const char* interfaceName,
char assignedInterfaceName[TUNInterface_IFNAMSIZ],
struct EventBase* base,
struct Log* logger,
struct Except* eh,
struct Allocator* alloc)
{
uint32_t maxNameSize = (IFNAMSIZ < TUNInterface_IFNAMSIZ) ? IFNAMSIZ : TUNInterface_IFNAMSIZ;
Log_info(logger, "Initializing tun device [%s]", ((interfaceName) ? interfaceName : "auto"));
struct ifreq ifRequest = { .ifr_flags = IFF_TUN };
if (interfaceName) {
if (strlen(interfaceName) > maxNameSize) {
Except_throw(eh, "tunnel name too big, limit is [%d] characters", maxNameSize);
}
strncpy(ifRequest.ifr_name, interfaceName, maxNameSize);
}
int fileno = open("/dev/net/tun", O_RDWR);
if (fileno < 0) {
Except_throw(eh, "open(\"/dev/net/tun\") [%s]", strerror(errno));
}
if (ioctl(fileno, TUNSETIFF, &ifRequest) < 0) {
int err = errno;
close(fileno);
Except_throw(eh, "ioctl(TUNSETIFF) [%s]", strerror(err));
}
strncpy(assignedInterfaceName, ifRequest.ifr_name, maxNameSize);
struct Pipe* p = Pipe_forFiles(fileno, fileno, base, eh, alloc);
return &p->iface;
}