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