/* 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 "interface/tuntap/TUNInterface_pvt.h"
#include "exception/Err.h"
#include "memory/Allocator.h"
#include "util/CString.h"
#include "util/platform/Socket.h"
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#if defined(Cjdns_android)
#define DEVICE_PATH "/dev/tun"
#else
#define DEVICE_PATH "/dev/net/tun"
#endif
#include
Err_DEFUN TUNInterface_newImpl(
Rffi_SocketIface_t** sout,
struct Iface** out,
const char* interfaceName,
char assignedInterfaceName[TUNInterface_IFNAMSIZ],
struct Log* logger,
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) {
Err_raise(alloc, "tunnel name too big, limit is [%d] characters", maxNameSize);
}
CString_safeStrncpy(ifRequest.ifr_name, interfaceName, maxNameSize);
}
int tunFd = open(DEVICE_PATH, O_RDWR);
if (tunFd < 0) {
Err_raise(alloc, "open(\"%s\") [%s]", DEVICE_PATH, strerror(errno));
}
if (ioctl(tunFd, TUNSETIFF, &ifRequest) < 0) {
int err = errno;
close(tunFd);
Err_raise(alloc, "ioctl(TUNSETIFF) [%s]", strerror(err));
}
if (assignedInterfaceName) {
CString_safeStrncpy(assignedInterfaceName, ifRequest.ifr_name, maxNameSize);
}
Socket_makeNonBlocking(tunFd);
return Rffi_socketForFd(out, sout, tunFd, RTypes_SocketType_ReadFrames, alloc);
}