#include <sys/stat.h>
#include <sys/syscall.h>
+#include <net/if.h>
+#include <net/if_arp.h>
+
+#include <arpa/inet.h>
+#include <netinet/in.h>
+
#include <linux/rtnetlink.h>
#include <linux/sockios.h>
+#include <linux/ip.h>
#include <linux/if_vlan.h>
#include <linux/if_bridge.h>
+#include <linux/if_tunnel.h>
#include <linux/ethtool.h>
#include <unistd.h>
static int cb_rtnl_event(struct nl_msg *msg, void *arg);
static void handle_hotplug_event(struct uloop_fd *u, unsigned int events);
+static char dev_buf[256];
+
static void
handler_nl_event(struct uloop_fd *u, unsigned int events)
{
static void system_set_dev_sysctl(const char *path, const char *device, const char *val)
{
- static char buf[256];
-
- snprintf(buf, sizeof(buf), path, val);
- system_set_sysctl(buf, val);
+ snprintf(dev_buf, sizeof(dev_buf), path, val);
+ system_set_sysctl(dev_buf, val);
}
static void system_set_disable_ipv6(struct device *dev, const char *val)
return ioctl(sock_ioctl, cmd, &ifr);
}
-int system_bridge_addif(struct device *bridge, struct device *dev)
-{
- system_set_disable_ipv6(dev, "1");
- return system_bridge_if(bridge->ifname, dev, SIOCBRADDIF, NULL);
-}
-
-int system_bridge_delif(struct device *bridge, struct device *dev)
-{
- system_set_disable_ipv6(dev, "0");
- return system_bridge_if(bridge->ifname, dev, SIOCBRDELIF, NULL);
-}
-
static bool system_is_bridge(const char *name, char *buf, int buflen)
{
struct stat st;
return path + 1;
}
+int system_bridge_addif(struct device *bridge, struct device *dev)
+{
+ char *oldbr;
+
+ system_set_disable_ipv6(dev, "1");
+ oldbr = system_get_bridge(dev->ifname, dev_buf, sizeof(dev_buf));
+ if (oldbr && !strcmp(oldbr, bridge->ifname))
+ return 0;
+
+ return system_bridge_if(bridge->ifname, dev, SIOCBRADDIF, NULL);
+}
+
+int system_bridge_delif(struct device *bridge, struct device *dev)
+{
+ system_set_disable_ipv6(dev, "0");
+ return system_bridge_if(bridge->ifname, dev, SIOCBRDELIF, NULL);
+}
+
static int system_if_resolve(struct device *dev)
{
struct ifreq ifr;
}
if (ioctl(sock_ioctl, SIOCGIFHWADDR, &ifr) == 0) {
- memcpy(s->macaddr, &ifr.ifr_hwaddr, sizeof(s->macaddr));
+ memcpy(s->macaddr, &ifr.ifr_hwaddr.sa_data, sizeof(s->macaddr));
s->flags |= DEV_OPT_MACADDR;
}
}
s->flags &= ~DEV_OPT_TXQUEUELEN;
}
if (s->flags & DEV_OPT_MACADDR) {
- memcpy(&ifr.ifr_hwaddr, s->macaddr, sizeof(s->macaddr));
+ ifr.ifr_hwaddr.sa_family = ARPHRD_ETHER;
+ memcpy(&ifr.ifr_hwaddr.sa_data, s->macaddr, sizeof(s->macaddr));
if (ioctl(sock_ioctl, SIOCSIFHWADDR, &ifr) < 0)
s->flags &= ~DEV_OPT_MACADDR;
}
int system_if_down(struct device *dev)
{
int ret = system_if_flags(dev->ifname, 0, IFF_UP);
+ dev->orig_settings.flags &= dev->settings.flags;
system_if_apply_settings(dev, &dev->orig_settings);
return ret;
}
static int system_addr(struct device *dev, struct device_addr *addr, int cmd)
{
- int alen = ((addr->flags & DEVADDR_FAMILY) == DEVADDR_INET4) ? 4 : 16;
+ bool v4 = ((addr->flags & DEVADDR_FAMILY) == DEVADDR_INET4);
+ int alen = v4 ? 4 : 16;
struct ifaddrmsg ifa = {
.ifa_family = (alen == 4) ? AF_INET : AF_INET6,
.ifa_prefixlen = addr->mask,
struct nl_msg *msg;
- dev = addr->device;
- if (dev) {
- if (!dev->ifindex)
- return -1;
-
- ifa.ifa_index = dev->ifindex;
- }
-
msg = nlmsg_alloc_simple(cmd, 0);
if (!msg)
return -1;
nlmsg_append(msg, &ifa, sizeof(ifa), 0);
nla_put(msg, IFA_LOCAL, alen, &addr->addr);
+ if (v4)
+ nla_put_u32(msg, IFA_BROADCAST, addr->broadcast);
+
return system_rtnl_call(msg);
}
if (cmd == RTM_NEWROUTE)
flags |= NLM_F_CREATE | NLM_F_REPLACE;
- dev = route->device;
- if (dev) {
- if (!dev->ifindex)
- return -1;
-
- ifindex = dev->ifindex;
- }
-
msg = nlmsg_alloc_simple(cmd, flags);
if (!msg)
return -1;
if (route->mask)
nla_put(msg, RTA_DST, alen, &route->addr);
- if (route->metric >= 0)
+ if (route->metric > 0)
nla_put_u32(msg, RTA_PRIORITY, route->metric);
if (have_gw)
return 0;
}
+
+#ifndef IP_DF
+#define IP_DF 0x4000
+#endif
+
+static void tunnel_parm_init(struct ip_tunnel_parm *p)
+{
+ memset(p, 0, sizeof(*p));
+ p->iph.version = 4;
+ p->iph.ihl = 5;
+ p->iph.frag_off = htons(IP_DF);
+}
+
+static int tunnel_ioctl(const char *name, int cmd, void *p)
+{
+ struct ifreq ifr;
+
+ memset(&ifr, 0, sizeof(ifr));
+ strncpy(ifr.ifr_name, name, sizeof(ifr.ifr_name));
+ ifr.ifr_ifru.ifru_data = p;
+ return ioctl(sock_ioctl, cmd, &ifr);
+}
+
+int system_del_ip_tunnel(const char *name)
+{
+ struct ip_tunnel_parm p;
+
+ tunnel_parm_init(&p);
+ return tunnel_ioctl(name, SIOCDELTUNNEL, &p);
+}
+
+static int parse_ipaddr(struct blob_attr *attr, __be32 *addr)
+{
+ if (!attr)
+ return 1;
+
+ return inet_pton(AF_INET, blobmsg_data(attr), (void *) addr);
+}
+
+
+int system_add_ip_tunnel(const char *name, struct blob_attr *attr)
+{
+ struct blob_attr *tb[__TUNNEL_ATTR_MAX];
+ struct blob_attr *cur;
+ struct ip_tunnel_parm p;
+ const char *base, *str;
+ int cmd = SIOCADDTUNNEL;
+
+ system_del_ip_tunnel(name);
+
+ tunnel_parm_init(&p);
+
+ blobmsg_parse(tunnel_attr_list.params, __TUNNEL_ATTR_MAX, tb,
+ blob_data(attr), blob_len(attr));
+
+ cur = tb[TUNNEL_ATTR_TYPE];
+ if (!cur)
+ return -EINVAL;
+
+ str = blobmsg_data(cur);
+ if (!strcmp(str, "sit")) {
+ p.iph.protocol = IPPROTO_IPV6;
+ base = "sit0";
+ } else
+ return -EINVAL;
+
+ if (!parse_ipaddr(tb[TUNNEL_ATTR_LOCAL], &p.iph.saddr))
+ return -EINVAL;
+
+ if (!parse_ipaddr(tb[TUNNEL_ATTR_REMOTE], &p.iph.daddr))
+ return -EINVAL;
+
+ if ((cur = tb[TUNNEL_ATTR_TTL])) {
+ unsigned int val = blobmsg_get_u32(cur);
+
+ if (val > 255)
+ return -EINVAL;
+
+ p.iph.ttl = val;
+ }
+
+ strncpy(p.name, name, sizeof(p.name));
+ return tunnel_ioctl(base, cmd, &p);
+}