X-Git-Url: http://git.archive.openwrt.org/?p=project%2Fnetifd.git;a=blobdiff_plain;f=system-linux.c;h=c4d094d21daf47ddce8f4a7950245ce2b9369ab7;hp=c7b8b8f66c8c45bdcdf3dbdd7f57de9376d8402b;hb=515a80251f14d4c033fd49918e9ecac308e135dd;hpb=575b1fe1bc8f15e65d89a0905e1640277c17fadc diff --git a/system-linux.c b/system-linux.c index c7b8b8f..c4d094d 100644 --- a/system-linux.c +++ b/system-linux.c @@ -769,6 +769,16 @@ static void system_add_link_modes(struct blob_buf *b, __u32 mask) } } +bool +system_if_force_external(const char *ifname) +{ + char buf[64]; + struct stat s; + + snprintf(buf, sizeof(buf), "/sys/class/net/%s/phy80211", ifname); + return stat(buf, &s) == 0; +} + int system_if_dump_info(struct device *dev, struct blob_buf *b) { @@ -1016,7 +1026,7 @@ int system_add_ip_tunnel(const char *name, struct blob_attr *attr) struct blob_attr *cur; struct ip_tunnel_parm p; const char *base, *str; - int cmd = SIOCADDTUNNEL; + bool is_sit; system_del_ip_tunnel(name); @@ -1025,12 +1035,12 @@ int system_add_ip_tunnel(const char *name, struct blob_attr *attr) blobmsg_parse(tunnel_attr_list.params, __TUNNEL_ATTR_MAX, tb, blob_data(attr), blob_len(attr)); - cur = tb[TUNNEL_ATTR_TYPE]; - if (!cur) + if (!(cur = tb[TUNNEL_ATTR_TYPE])) return -EINVAL; - str = blobmsg_data(cur); - if (!strcmp(str, "sit")) { + is_sit = !strcmp(str, "sit"); + + if (is_sit) { p.iph.protocol = IPPROTO_IPV6; base = "sit0"; } else @@ -1052,5 +1062,35 @@ int system_add_ip_tunnel(const char *name, struct blob_attr *attr) } strncpy(p.name, name, sizeof(p.name)); - return tunnel_ioctl(base, cmd, &p); + if (tunnel_ioctl(base, SIOCADDTUNNEL, &p) < 0) + return -1; + +#ifdef SIOCADD6RD + cur = tb[TUNNEL_ATTR_6RD_PREFIX]; + if (cur && is_sit) { + unsigned int mask; + struct ip_tunnel_6rd p6; + + memset(&p6, 0, sizeof(p6)); + + if (!parse_ip_and_netmask(AF_INET6, blobmsg_data(cur), + &p6.prefix, &mask) || mask > 128) + return -EINVAL; + p6.prefixlen = mask; + + if ((cur = tb[TUNNEL_ATTR_6RD_RELAY_PREFIX])) { + if (!parse_ip_and_netmask(AF_INET, blobmsg_data(cur), + &p6.relay_prefix, &mask) || mask > 32) + return -EINVAL; + p6.relay_prefixlen = mask; + } + + if (tunnel_ioctl(name, SIOCADD6RD, &p6) < 0) { + system_del_ip_tunnel(name); + return -1; + } + } +#endif + + return 0; }