Rename "subsystems" directory to "base"
[oonf.git] / src / base / os_linux / os_vif_linux.c
1
2 /*
3  * The olsr.org Optimized Link-State Routing daemon version 2 (olsrd2)
4  * Copyright (c) 2004-2015, the olsr.org team - see HISTORY file
5  * All rights reserved.
6  *
7  * Redistribution and use in source and binary forms, with or without
8  * modification, are permitted provided that the following conditions
9  * are met:
10  *
11  * * Redistributions of source code must retain the above copyright
12  *   notice, this list of conditions and the following disclaimer.
13  * * Redistributions in binary form must reproduce the above copyright
14  *   notice, this list of conditions and the following disclaimer in
15  *   the documentation and/or other materials provided with the
16  *   distribution.
17  * * Neither the name of olsr.org, olsrd nor the names of its
18  *   contributors may be used to endorse or promote products derived
19  *   from this software without specific prior written permission.
20  *
21  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
22  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
23  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
24  * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
25  * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
26  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
27  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
28  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
29  * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
30  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
31  * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
32  * POSSIBILITY OF SUCH DAMAGE.
33  *
34  * Visit http://www.olsr.org for more information.
35  *
36  * If you find this software useful feel free to make a donation
37  * to the project. For more information see the website or contact
38  * the copyright holders.
39  *
40  */
41
42 /**
43  * @file
44  */
45
46 #include <errno.h>
47 #include <fcntl.h>
48 #include <linux/if_tun.h>
49 #include <sys/ioctl.h>
50 #include <sys/stat.h>
51 #include <sys/types.h>
52 #include <unistd.h>
53
54 #include <oonf/libcommon/avl.h>
55 #include <oonf/libcommon/avl_comp.h>
56 #include <oonf/oonf.h>
57 #include <oonf/libcore/oonf_subsystem.h>
58 #include <oonf/base/os_vif.h>
59
60 /* Definitions */
61 #define LOG_OS_VIF _oonf_os_vif_subsystem.logging
62
63 static int _init(void);
64 static void _cleanup(void);
65
66 /* subsystem definition */
67 static const char *_dependencies[0] = {};
68
69 static struct oonf_subsystem _oonf_os_vif_subsystem = {
70   .name = OONF_OS_VIF_SUBSYSTEM,
71   .dependencies = _dependencies,
72   .dependencies_count = ARRAYSIZE(_dependencies),
73   .init = _init,
74   .cleanup = _cleanup,
75 };
76 DECLARE_OONF_PLUGIN(_oonf_os_vif_subsystem);
77
78 static struct avl_tree _vif_tree;
79
80 /**
81  * Initialize virtual interface subsystem
82  * @return -1 if an error happened, 0 otherwise
83  */
84 static int
85 _init(void) {
86   avl_init(&_vif_tree, avl_comp_strcasecmp, false);
87   return 0;
88 }
89
90 /**
91  * Cleanup virtual interface subsystem
92  */
93 static void
94 _cleanup(void) {
95   struct os_vif *vif, *vif_it;
96
97   avl_for_each_element_safe(&_vif_tree, vif, _vif_node, vif_it) {
98     os_vif_close(vif);
99   }
100 }
101
102 /**
103  * Open a new virtual interface
104  * @param sock os socket
105  * @param vif pointer to virtual interface object
106  * @return -1 if an error happened, 0 otherwise
107  */
108 int
109 os_vif_linux_open(struct os_fd *sock, struct os_vif *vif) {
110   struct ifreq if_req;
111   int fd, flag;
112
113   switch (vif->type) {
114     case OS_VIF_MAC:
115       flag = IFF_TAP;
116       break;
117     case OS_VIF_IP:
118       flag = IFF_TUN;
119       break;
120     default:
121       OONF_WARN(LOG_OS_VIF, "Unknown vif type: %d", vif->type);
122       return -1;
123   }
124
125   fd = open("/dev/net/tun", O_RDWR);
126   if (fd < 0) {
127     OONF_WARN(LOG_OS_VIF, "Cannot open virtual interface device: %s (%d)", strerror(errno), errno);
128     return -1;
129   }
130
131   memset(&if_req, 0, sizeof(if_req));
132   strscpy(if_req.ifr_name, vif->if_name, IF_NAMESIZE);
133
134   /*
135    * Specify the IFF_TAP flag for Ethernet packets.
136    * Specify IFF_NO_PI for not receiving extra meta packet information.
137    */
138   if_req.ifr_flags = flag | IFF_NO_PI;
139
140   if (ioctl(fd, TUNSETIFF, (void *)&if_req) < 0) {
141     OONF_WARN(LOG_OS_VIF, "Cannot set mode of virtual interface device: %s (%d)", strerror(errno), errno);
142     close(fd);
143     return -1;
144   }
145
146   /* initialize vif memory */
147   os_fd_init(sock, fd);
148   vif->_vif_node.key = vif->if_name;
149
150   avl_insert(&_vif_tree, &vif->_vif_node);
151
152   return 0;
153 }
154
155 /**
156  * Close a virtual interface
157  * @param vif pointer to virtual interface object
158  */
159 void
160 os_vif_linux_close(struct os_vif *vif) {
161   if (avl_is_node_added(&vif->_vif_node)) {
162     os_fd_close(vif->fd);
163     avl_remove(&_vif_tree, &vif->_vif_node);
164   }
165 }
166
167 /**
168  * get virtual interface tree
169  * @return vif tree
170  */
171 struct avl_tree *
172 os_vif_linux_get_tree(void) {
173   return &_vif_tree;
174 }