/* * * Connection Manager * * Copyright (C) 2007-2008 Intel Corporation. All rights reserved. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 as * published by the Free Software Foundation. * * 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, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA * */ #ifdef HAVE_CONFIG_H #include #endif #include #include #include #include #include #include #include #include #include #include #include struct ethernet_data { int index; unsigned flags; }; static GSList *ethernet_list = NULL; static void ethernet_newlink(unsigned short type, int index, unsigned flags, unsigned change) { GSList *list; DBG("index %d flags %ld change %ld", index, flags, change); for (list = ethernet_list; list; list = list->next) { struct connman_element *element = list->data; struct connman_element *netdev; struct ethernet_data *ethernet; ethernet = connman_element_get_data(element); if (ethernet == NULL) continue; if (ethernet->index != index) continue; if ((ethernet->flags & IFF_RUNNING) == (flags & IFF_RUNNING)) continue; ethernet->flags = flags; if (ethernet->flags & IFF_RUNNING) { DBG("carrier on"); netdev = connman_element_create(NULL); if (netdev != NULL) { netdev->type = CONNMAN_ELEMENT_TYPE_DEVICE; netdev->subtype = CONNMAN_ELEMENT_SUBTYPE_NETWORK; netdev->index = element->index; connman_element_register(netdev, element); } } else { DBG("carrier off"); connman_element_unregister_children(element); } } } static struct connman_rtnl ethernet_rtnl = { .name = "ethernet", .newlink = ethernet_newlink, }; static int iface_up(struct ethernet_data *ethernet) { struct ifreq ifr; int sk, err; DBG("index %d flags %d", ethernet->index, ethernet->flags); sk = socket(PF_INET, SOCK_DGRAM, 0); if (sk < 0) return -errno; memset(&ifr, 0, sizeof(ifr)); ifr.ifr_ifindex = ethernet->index; if (ioctl(sk, SIOCGIFNAME, &ifr) < 0) { err = -errno; goto done; } if (ioctl(sk, SIOCGIFFLAGS, &ifr) < 0) { err = -errno; goto done; } if (ifr.ifr_flags & IFF_UP) { err = -EALREADY; goto done; } ifr.ifr_flags |= IFF_UP; if (ioctl(sk, SIOCSIFFLAGS, &ifr) < 0) { err = -errno; goto done; } err = 0; done: close(sk); return err; } static int iface_down(struct ethernet_data *ethernet) { struct ifreq ifr; int sk, err; DBG("index %d flags %d", ethernet->index, ethernet->flags); sk = socket(PF_INET, SOCK_DGRAM, 0); if (sk < 0) return -errno; memset(&ifr, 0, sizeof(ifr)); ifr.ifr_ifindex = ethernet->index; if (ioctl(sk, SIOCGIFNAME, &ifr) < 0) { err = -errno; goto done; } if (ioctl(sk, SIOCGIFFLAGS, &ifr) < 0) { err = -errno; goto done; } if (!(ifr.ifr_flags & IFF_UP)) { err = -EALREADY; goto done; } ifr.ifr_flags &= ~IFF_UP; if (ioctl(sk, SIOCSIFFLAGS, &ifr) < 0) err = -errno; else err = 0; done: close(sk); return err; } static int ethernet_probe(struct connman_element *element) { struct ethernet_data *ethernet; DBG("element %p name %s", element, element->name); ethernet = g_try_new0(struct ethernet_data, 1); if (ethernet == NULL) return -ENOMEM; ethernet_list = g_slist_append(ethernet_list, element); connman_element_set_data(element, ethernet); ethernet->index = element->index; connman_rtnl_send_getlink(); return 0; } static void ethernet_remove(struct connman_element *element) { struct ethernet_data *ethernet = connman_element_get_data(element); DBG("element %p name %s", element, element->name); connman_element_set_data(element, NULL); ethernet_list = g_slist_remove(ethernet_list, element); g_free(ethernet); } static int ethernet_enable(struct connman_element *element) { struct ethernet_data *ethernet = connman_element_get_data(element); DBG("element %p name %s", element, element->name); iface_up(ethernet); return 0; } static int ethernet_disable(struct connman_element *element) { struct ethernet_data *ethernet = connman_element_get_data(element); DBG("element %p name %s", element, element->name); iface_down(ethernet); return 0; } static struct connman_driver ethernet_driver = { .name = "ethernet", .type = CONNMAN_ELEMENT_TYPE_DEVICE, .subtype = CONNMAN_ELEMENT_SUBTYPE_ETHERNET, .probe = ethernet_probe, .remove = ethernet_remove, .enable = ethernet_enable, .disable = ethernet_disable, }; static int ethernet_init(void) { int err; err = connman_rtnl_register(ðernet_rtnl); if (err < 0) return err; err = connman_driver_register(ðernet_driver); if (err < 0) { connman_rtnl_unregister(ðernet_rtnl); return err; } return 0; } static void ethernet_exit(void) { connman_driver_unregister(ðernet_driver); connman_rtnl_unregister(ðernet_rtnl); } CONNMAN_PLUGIN_DEFINE("ethernet", "Ethernet interface plugin", VERSION, ethernet_init, ethernet_exit)