5 * Copyright (C) 2007-2012 Intel Corporation. All rights reserved.
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License version 2 as
9 * published by the Free Software Foundation.
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
30 #define IFF_LOWER_UP 0x10000
35 #define CONNMAN_API_SUBJECT_TO_CHANGE
36 #include <connman/technology.h>
37 #include <connman/plugin.h>
38 #include <connman/device.h>
39 #include <connman/inet.h>
40 #include <connman/rtnl.h>
41 #include <connman/log.h>
43 struct ethernet_data {
47 struct connman_network *network;
50 static int cable_probe(struct connman_network *network)
52 DBG("network %p", network);
57 static void cable_remove(struct connman_network *network)
59 DBG("network %p", network);
62 static int cable_connect(struct connman_network *network)
64 DBG("network %p", network);
66 connman_network_set_connected(network, TRUE);
71 static int cable_disconnect(struct connman_network *network)
73 DBG("network %p", network);
75 connman_network_set_connected(network, FALSE);
80 static struct connman_network_driver cable_driver = {
82 .type = CONNMAN_NETWORK_TYPE_ETHERNET,
84 .remove = cable_remove,
85 .connect = cable_connect,
86 .disconnect = cable_disconnect,
89 static void add_network(struct connman_device *device,
90 struct ethernet_data *ethernet)
92 struct connman_network *network;
95 network = connman_network_create("carrier",
96 CONNMAN_NETWORK_TYPE_ETHERNET);
100 index = connman_device_get_index(device);
101 connman_network_set_index(network, index);
103 connman_network_set_name(network, "Wired");
105 if (connman_device_add_network(device, network) < 0) {
106 connman_network_unref(network);
110 connman_network_set_available(network, TRUE);
112 connman_network_set_group(network, "cable");
114 ethernet->network = network;
117 static void remove_network(struct connman_device *device,
118 struct ethernet_data *ethernet)
120 if (ethernet->network == NULL)
123 connman_device_remove_network(device, ethernet->network);
124 connman_network_unref(ethernet->network);
126 ethernet->network = NULL;
129 static void ethernet_newlink(unsigned flags, unsigned change, void *user_data)
131 struct connman_device *device = user_data;
132 struct ethernet_data *ethernet = connman_device_get_data(device);
134 DBG("index %d flags %d change %d", ethernet->index, flags, change);
136 if ((ethernet->flags & IFF_UP) != (flags & IFF_UP)) {
137 if (flags & IFF_UP) {
139 connman_device_set_powered(device, TRUE);
142 connman_device_set_powered(device, FALSE);
146 if ((ethernet->flags & IFF_LOWER_UP) != (flags & IFF_LOWER_UP)) {
147 if (flags & IFF_LOWER_UP) {
149 add_network(device, ethernet);
152 remove_network(device, ethernet);
156 ethernet->flags = flags;
159 static int ethernet_probe(struct connman_device *device)
161 struct ethernet_data *ethernet;
163 DBG("device %p", device);
165 ethernet = g_try_new0(struct ethernet_data, 1);
166 if (ethernet == NULL)
169 connman_device_set_data(device, ethernet);
171 ethernet->index = connman_device_get_index(device);
174 ethernet->watch = connman_rtnl_add_newlink_watch(ethernet->index,
175 ethernet_newlink, device);
180 static void ethernet_remove(struct connman_device *device)
182 struct ethernet_data *ethernet = connman_device_get_data(device);
184 DBG("device %p", device);
186 connman_device_set_data(device, NULL);
188 connman_rtnl_remove_watch(ethernet->watch);
190 remove_network(device, ethernet);
195 static int ethernet_enable(struct connman_device *device)
197 struct ethernet_data *ethernet = connman_device_get_data(device);
199 DBG("device %p", device);
201 return connman_inet_ifup(ethernet->index);
204 static int ethernet_disable(struct connman_device *device)
206 struct ethernet_data *ethernet = connman_device_get_data(device);
208 DBG("device %p", device);
210 return connman_inet_ifdown(ethernet->index);
213 static struct connman_device_driver ethernet_driver = {
215 .type = CONNMAN_DEVICE_TYPE_ETHERNET,
216 .probe = ethernet_probe,
217 .remove = ethernet_remove,
218 .enable = ethernet_enable,
219 .disable = ethernet_disable,
222 static GList *cdc_interface_list = NULL;
224 static void tech_add_interface(struct connman_technology *technology,
225 int index, const char *name, const char *ident)
227 DBG("index %d name %s ident %s", index, name, ident);
229 if (g_list_find(cdc_interface_list,
230 GINT_TO_POINTER((int) index)) != NULL)
233 cdc_interface_list = g_list_prepend(cdc_interface_list,
234 (GINT_TO_POINTER((int) index)));
237 static void tech_remove_interface(struct connman_technology *technology,
240 DBG("index %d", index);
242 cdc_interface_list = g_list_remove(cdc_interface_list,
243 GINT_TO_POINTER((int) index));
246 static void enable_tethering(struct connman_technology *technology,
251 for (list = cdc_interface_list; list; list = list->next) {
252 int index = GPOINTER_TO_INT(list->data);
254 connman_technology_tethering_notify(technology, TRUE);
256 connman_inet_ifup(index);
258 connman_inet_add_to_bridge(index, bridge);
262 static void disable_tethering(struct connman_technology *technology,
267 for (list = cdc_interface_list; list; list = list->next) {
268 int index = GPOINTER_TO_INT(list->data);
270 connman_inet_remove_from_bridge(index, bridge);
272 connman_inet_ifdown(index);
274 connman_technology_tethering_notify(technology, FALSE);
278 static int tech_set_tethering(struct connman_technology *technology,
279 const char *identifier, const char *passphrase,
280 const char *bridge, connman_bool_t enabled)
282 DBG("bridge %s enabled %d", bridge, enabled);
285 enable_tethering(technology, bridge);
287 disable_tethering(technology, bridge);
292 static int tech_probe(struct connman_technology *technology)
297 static void tech_remove(struct connman_technology *technology)
299 g_list_free(cdc_interface_list);
301 cdc_interface_list = NULL;
304 static struct connman_technology_driver tech_driver = {
305 .name = "cdc_ethernet",
306 .type = CONNMAN_SERVICE_TYPE_GADGET,
308 .remove = tech_remove,
309 .add_interface = tech_add_interface,
310 .remove_interface = tech_remove_interface,
311 .set_tethering = tech_set_tethering,
314 static int eth_probe(struct connman_technology *technology)
319 static void eth_remove(struct connman_technology *technology)
324 static struct connman_technology_driver eth_driver = {
326 .type = CONNMAN_SERVICE_TYPE_ETHERNET,
328 .remove = eth_remove,
331 static int ethernet_init(void)
335 err = connman_technology_driver_register(ð_driver);
339 err = connman_network_driver_register(&cable_driver);
343 err = connman_device_driver_register(ðernet_driver);
345 connman_network_driver_unregister(&cable_driver);
349 err = connman_technology_driver_register(&tech_driver);
351 connman_device_driver_unregister(ðernet_driver);
352 connman_network_driver_unregister(&cable_driver);
359 static void ethernet_exit(void)
361 connman_technology_driver_unregister(ð_driver);
363 connman_technology_driver_unregister(&tech_driver);
365 connman_network_driver_unregister(&cable_driver);
367 connman_device_driver_unregister(ðernet_driver);
370 CONNMAN_PLUGIN_DEFINE(ethernet, "Ethernet interface plugin", VERSION,
371 CONNMAN_PLUGIN_PRIORITY_DEFAULT, ethernet_init, ethernet_exit)