|
@@ -546,15 +546,14 @@ struct rt6_info *rt6_lookup(struct in6_addr *daddr, struct in6_addr *saddr,
|
|
be destroyed.
|
|
be destroyed.
|
|
*/
|
|
*/
|
|
|
|
|
|
-static int __ip6_ins_rt(struct rt6_info *rt, struct nlmsghdr *nlh,
|
|
|
|
- void *_rtattr, struct netlink_skb_parms *req)
|
|
|
|
|
|
+static int __ip6_ins_rt(struct rt6_info *rt, struct nl_info *info)
|
|
{
|
|
{
|
|
int err;
|
|
int err;
|
|
struct fib6_table *table;
|
|
struct fib6_table *table;
|
|
|
|
|
|
table = rt->rt6i_table;
|
|
table = rt->rt6i_table;
|
|
write_lock_bh(&table->tb6_lock);
|
|
write_lock_bh(&table->tb6_lock);
|
|
- err = fib6_add(&table->tb6_root, rt, nlh, _rtattr, req);
|
|
|
|
|
|
+ err = fib6_add(&table->tb6_root, rt, info);
|
|
write_unlock_bh(&table->tb6_lock);
|
|
write_unlock_bh(&table->tb6_lock);
|
|
|
|
|
|
return err;
|
|
return err;
|
|
@@ -562,7 +561,7 @@ static int __ip6_ins_rt(struct rt6_info *rt, struct nlmsghdr *nlh,
|
|
|
|
|
|
int ip6_ins_rt(struct rt6_info *rt)
|
|
int ip6_ins_rt(struct rt6_info *rt)
|
|
{
|
|
{
|
|
- return __ip6_ins_rt(rt, NULL, NULL, NULL);
|
|
|
|
|
|
+ return __ip6_ins_rt(rt, NULL);
|
|
}
|
|
}
|
|
|
|
|
|
static struct rt6_info *rt6_alloc_cow(struct rt6_info *ort, struct in6_addr *daddr,
|
|
static struct rt6_info *rt6_alloc_cow(struct rt6_info *ort, struct in6_addr *daddr,
|
|
@@ -1014,30 +1013,24 @@ int ipv6_get_hoplimit(struct net_device *dev)
|
|
*
|
|
*
|
|
*/
|
|
*/
|
|
|
|
|
|
-int ip6_route_add(struct in6_rtmsg *rtmsg, struct nlmsghdr *nlh,
|
|
|
|
- void *_rtattr, struct netlink_skb_parms *req,
|
|
|
|
- u32 table_id)
|
|
|
|
|
|
+int ip6_route_add(struct fib6_config *cfg)
|
|
{
|
|
{
|
|
int err;
|
|
int err;
|
|
- struct rtmsg *r;
|
|
|
|
- struct rtattr **rta;
|
|
|
|
struct rt6_info *rt = NULL;
|
|
struct rt6_info *rt = NULL;
|
|
struct net_device *dev = NULL;
|
|
struct net_device *dev = NULL;
|
|
struct inet6_dev *idev = NULL;
|
|
struct inet6_dev *idev = NULL;
|
|
struct fib6_table *table;
|
|
struct fib6_table *table;
|
|
int addr_type;
|
|
int addr_type;
|
|
|
|
|
|
- rta = (struct rtattr **) _rtattr;
|
|
|
|
-
|
|
|
|
- if (rtmsg->rtmsg_dst_len > 128 || rtmsg->rtmsg_src_len > 128)
|
|
|
|
|
|
+ if (cfg->fc_dst_len > 128 || cfg->fc_src_len > 128)
|
|
return -EINVAL;
|
|
return -EINVAL;
|
|
#ifndef CONFIG_IPV6_SUBTREES
|
|
#ifndef CONFIG_IPV6_SUBTREES
|
|
- if (rtmsg->rtmsg_src_len)
|
|
|
|
|
|
+ if (cfg->fc_src_len)
|
|
return -EINVAL;
|
|
return -EINVAL;
|
|
#endif
|
|
#endif
|
|
- if (rtmsg->rtmsg_ifindex) {
|
|
|
|
|
|
+ if (cfg->fc_ifindex) {
|
|
err = -ENODEV;
|
|
err = -ENODEV;
|
|
- dev = dev_get_by_index(rtmsg->rtmsg_ifindex);
|
|
|
|
|
|
+ dev = dev_get_by_index(cfg->fc_ifindex);
|
|
if (!dev)
|
|
if (!dev)
|
|
goto out;
|
|
goto out;
|
|
idev = in6_dev_get(dev);
|
|
idev = in6_dev_get(dev);
|
|
@@ -1045,10 +1038,10 @@ int ip6_route_add(struct in6_rtmsg *rtmsg, struct nlmsghdr *nlh,
|
|
goto out;
|
|
goto out;
|
|
}
|
|
}
|
|
|
|
|
|
- if (rtmsg->rtmsg_metric == 0)
|
|
|
|
- rtmsg->rtmsg_metric = IP6_RT_PRIO_USER;
|
|
|
|
|
|
+ if (cfg->fc_metric == 0)
|
|
|
|
+ cfg->fc_metric = IP6_RT_PRIO_USER;
|
|
|
|
|
|
- table = fib6_new_table(table_id);
|
|
|
|
|
|
+ table = fib6_new_table(cfg->fc_table);
|
|
if (table == NULL) {
|
|
if (table == NULL) {
|
|
err = -ENOBUFS;
|
|
err = -ENOBUFS;
|
|
goto out;
|
|
goto out;
|
|
@@ -1062,14 +1055,13 @@ int ip6_route_add(struct in6_rtmsg *rtmsg, struct nlmsghdr *nlh,
|
|
}
|
|
}
|
|
|
|
|
|
rt->u.dst.obsolete = -1;
|
|
rt->u.dst.obsolete = -1;
|
|
- rt->rt6i_expires = jiffies + clock_t_to_jiffies(rtmsg->rtmsg_info);
|
|
|
|
- if (nlh && (r = NLMSG_DATA(nlh))) {
|
|
|
|
- rt->rt6i_protocol = r->rtm_protocol;
|
|
|
|
- } else {
|
|
|
|
- rt->rt6i_protocol = RTPROT_BOOT;
|
|
|
|
- }
|
|
|
|
|
|
+ rt->rt6i_expires = jiffies + clock_t_to_jiffies(cfg->fc_expires);
|
|
|
|
+
|
|
|
|
+ if (cfg->fc_protocol == RTPROT_UNSPEC)
|
|
|
|
+ cfg->fc_protocol = RTPROT_BOOT;
|
|
|
|
+ rt->rt6i_protocol = cfg->fc_protocol;
|
|
|
|
|
|
- addr_type = ipv6_addr_type(&rtmsg->rtmsg_dst);
|
|
|
|
|
|
+ addr_type = ipv6_addr_type(&cfg->fc_dst);
|
|
|
|
|
|
if (addr_type & IPV6_ADDR_MULTICAST)
|
|
if (addr_type & IPV6_ADDR_MULTICAST)
|
|
rt->u.dst.input = ip6_mc_input;
|
|
rt->u.dst.input = ip6_mc_input;
|
|
@@ -1078,24 +1070,22 @@ int ip6_route_add(struct in6_rtmsg *rtmsg, struct nlmsghdr *nlh,
|
|
|
|
|
|
rt->u.dst.output = ip6_output;
|
|
rt->u.dst.output = ip6_output;
|
|
|
|
|
|
- ipv6_addr_prefix(&rt->rt6i_dst.addr,
|
|
|
|
- &rtmsg->rtmsg_dst, rtmsg->rtmsg_dst_len);
|
|
|
|
- rt->rt6i_dst.plen = rtmsg->rtmsg_dst_len;
|
|
|
|
|
|
+ ipv6_addr_prefix(&rt->rt6i_dst.addr, &cfg->fc_dst, cfg->fc_dst_len);
|
|
|
|
+ rt->rt6i_dst.plen = cfg->fc_dst_len;
|
|
if (rt->rt6i_dst.plen == 128)
|
|
if (rt->rt6i_dst.plen == 128)
|
|
rt->u.dst.flags = DST_HOST;
|
|
rt->u.dst.flags = DST_HOST;
|
|
|
|
|
|
#ifdef CONFIG_IPV6_SUBTREES
|
|
#ifdef CONFIG_IPV6_SUBTREES
|
|
- ipv6_addr_prefix(&rt->rt6i_src.addr,
|
|
|
|
- &rtmsg->rtmsg_src, rtmsg->rtmsg_src_len);
|
|
|
|
- rt->rt6i_src.plen = rtmsg->rtmsg_src_len;
|
|
|
|
|
|
+ ipv6_addr_prefix(&rt->rt6i_src.addr, &cfg->fc_src, cfg->fc_src_len);
|
|
|
|
+ rt->rt6i_src.plen = cfg->fc_src_len;
|
|
#endif
|
|
#endif
|
|
|
|
|
|
- rt->rt6i_metric = rtmsg->rtmsg_metric;
|
|
|
|
|
|
+ rt->rt6i_metric = cfg->fc_metric;
|
|
|
|
|
|
/* We cannot add true routes via loopback here,
|
|
/* We cannot add true routes via loopback here,
|
|
they would result in kernel looping; promote them to reject routes
|
|
they would result in kernel looping; promote them to reject routes
|
|
*/
|
|
*/
|
|
- if ((rtmsg->rtmsg_flags&RTF_REJECT) ||
|
|
|
|
|
|
+ if ((cfg->fc_flags & RTF_REJECT) ||
|
|
(dev && (dev->flags&IFF_LOOPBACK) && !(addr_type&IPV6_ADDR_LOOPBACK))) {
|
|
(dev && (dev->flags&IFF_LOOPBACK) && !(addr_type&IPV6_ADDR_LOOPBACK))) {
|
|
/* hold loopback dev/idev if we haven't done so. */
|
|
/* hold loopback dev/idev if we haven't done so. */
|
|
if (dev != &loopback_dev) {
|
|
if (dev != &loopback_dev) {
|
|
@@ -1118,12 +1108,12 @@ int ip6_route_add(struct in6_rtmsg *rtmsg, struct nlmsghdr *nlh,
|
|
goto install_route;
|
|
goto install_route;
|
|
}
|
|
}
|
|
|
|
|
|
- if (rtmsg->rtmsg_flags & RTF_GATEWAY) {
|
|
|
|
|
|
+ if (cfg->fc_flags & RTF_GATEWAY) {
|
|
struct in6_addr *gw_addr;
|
|
struct in6_addr *gw_addr;
|
|
int gwa_type;
|
|
int gwa_type;
|
|
|
|
|
|
- gw_addr = &rtmsg->rtmsg_gateway;
|
|
|
|
- ipv6_addr_copy(&rt->rt6i_gateway, &rtmsg->rtmsg_gateway);
|
|
|
|
|
|
+ gw_addr = &cfg->fc_gateway;
|
|
|
|
+ ipv6_addr_copy(&rt->rt6i_gateway, gw_addr);
|
|
gwa_type = ipv6_addr_type(gw_addr);
|
|
gwa_type = ipv6_addr_type(gw_addr);
|
|
|
|
|
|
if (gwa_type != (IPV6_ADDR_LINKLOCAL|IPV6_ADDR_UNICAST)) {
|
|
if (gwa_type != (IPV6_ADDR_LINKLOCAL|IPV6_ADDR_UNICAST)) {
|
|
@@ -1140,7 +1130,7 @@ int ip6_route_add(struct in6_rtmsg *rtmsg, struct nlmsghdr *nlh,
|
|
if (!(gwa_type&IPV6_ADDR_UNICAST))
|
|
if (!(gwa_type&IPV6_ADDR_UNICAST))
|
|
goto out;
|
|
goto out;
|
|
|
|
|
|
- grt = rt6_lookup(gw_addr, NULL, rtmsg->rtmsg_ifindex, 1);
|
|
|
|
|
|
+ grt = rt6_lookup(gw_addr, NULL, cfg->fc_ifindex, 1);
|
|
|
|
|
|
err = -EHOSTUNREACH;
|
|
err = -EHOSTUNREACH;
|
|
if (grt == NULL)
|
|
if (grt == NULL)
|
|
@@ -1172,7 +1162,7 @@ int ip6_route_add(struct in6_rtmsg *rtmsg, struct nlmsghdr *nlh,
|
|
if (dev == NULL)
|
|
if (dev == NULL)
|
|
goto out;
|
|
goto out;
|
|
|
|
|
|
- if (rtmsg->rtmsg_flags & (RTF_GATEWAY|RTF_NONEXTHOP)) {
|
|
|
|
|
|
+ if (cfg->fc_flags & (RTF_GATEWAY | RTF_NONEXTHOP)) {
|
|
rt->rt6i_nexthop = __neigh_lookup_errno(&nd_tbl, &rt->rt6i_gateway, dev);
|
|
rt->rt6i_nexthop = __neigh_lookup_errno(&nd_tbl, &rt->rt6i_gateway, dev);
|
|
if (IS_ERR(rt->rt6i_nexthop)) {
|
|
if (IS_ERR(rt->rt6i_nexthop)) {
|
|
err = PTR_ERR(rt->rt6i_nexthop);
|
|
err = PTR_ERR(rt->rt6i_nexthop);
|
|
@@ -1181,24 +1171,24 @@ int ip6_route_add(struct in6_rtmsg *rtmsg, struct nlmsghdr *nlh,
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
- rt->rt6i_flags = rtmsg->rtmsg_flags;
|
|
|
|
|
|
+ rt->rt6i_flags = cfg->fc_flags;
|
|
|
|
|
|
install_route:
|
|
install_route:
|
|
- if (rta && rta[RTA_METRICS-1]) {
|
|
|
|
- int attrlen = RTA_PAYLOAD(rta[RTA_METRICS-1]);
|
|
|
|
- struct rtattr *attr = RTA_DATA(rta[RTA_METRICS-1]);
|
|
|
|
-
|
|
|
|
- while (RTA_OK(attr, attrlen)) {
|
|
|
|
- unsigned flavor = attr->rta_type;
|
|
|
|
- if (flavor) {
|
|
|
|
- if (flavor > RTAX_MAX) {
|
|
|
|
|
|
+ if (cfg->fc_mx) {
|
|
|
|
+ struct nlattr *nla;
|
|
|
|
+ int remaining;
|
|
|
|
+
|
|
|
|
+ nla_for_each_attr(nla, cfg->fc_mx, cfg->fc_mx_len, remaining) {
|
|
|
|
+ int type = nla->nla_type;
|
|
|
|
+
|
|
|
|
+ if (type) {
|
|
|
|
+ if (type > RTAX_MAX) {
|
|
err = -EINVAL;
|
|
err = -EINVAL;
|
|
goto out;
|
|
goto out;
|
|
}
|
|
}
|
|
- rt->u.dst.metrics[flavor-1] =
|
|
|
|
- *(u32 *)RTA_DATA(attr);
|
|
|
|
|
|
+
|
|
|
|
+ rt->u.dst.metrics[type - 1] = nla_get_u32(nla);
|
|
}
|
|
}
|
|
- attr = RTA_NEXT(attr, attrlen);
|
|
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
@@ -1211,7 +1201,7 @@ install_route:
|
|
rt->u.dst.dev = dev;
|
|
rt->u.dst.dev = dev;
|
|
rt->rt6i_idev = idev;
|
|
rt->rt6i_idev = idev;
|
|
rt->rt6i_table = table;
|
|
rt->rt6i_table = table;
|
|
- return __ip6_ins_rt(rt, nlh, _rtattr, req);
|
|
|
|
|
|
+ return __ip6_ins_rt(rt, &cfg->fc_nlinfo);
|
|
|
|
|
|
out:
|
|
out:
|
|
if (dev)
|
|
if (dev)
|
|
@@ -1223,8 +1213,7 @@ out:
|
|
return err;
|
|
return err;
|
|
}
|
|
}
|
|
|
|
|
|
-static int __ip6_del_rt(struct rt6_info *rt, struct nlmsghdr *nlh,
|
|
|
|
- void *_rtattr, struct netlink_skb_parms *req)
|
|
|
|
|
|
+static int __ip6_del_rt(struct rt6_info *rt, struct nl_info *info)
|
|
{
|
|
{
|
|
int err;
|
|
int err;
|
|
struct fib6_table *table;
|
|
struct fib6_table *table;
|
|
@@ -1235,7 +1224,7 @@ static int __ip6_del_rt(struct rt6_info *rt, struct nlmsghdr *nlh,
|
|
table = rt->rt6i_table;
|
|
table = rt->rt6i_table;
|
|
write_lock_bh(&table->tb6_lock);
|
|
write_lock_bh(&table->tb6_lock);
|
|
|
|
|
|
- err = fib6_del(rt, nlh, _rtattr, req);
|
|
|
|
|
|
+ err = fib6_del(rt, info);
|
|
dst_release(&rt->u.dst);
|
|
dst_release(&rt->u.dst);
|
|
|
|
|
|
write_unlock_bh(&table->tb6_lock);
|
|
write_unlock_bh(&table->tb6_lock);
|
|
@@ -1245,44 +1234,41 @@ static int __ip6_del_rt(struct rt6_info *rt, struct nlmsghdr *nlh,
|
|
|
|
|
|
int ip6_del_rt(struct rt6_info *rt)
|
|
int ip6_del_rt(struct rt6_info *rt)
|
|
{
|
|
{
|
|
- return __ip6_del_rt(rt, NULL, NULL, NULL);
|
|
|
|
|
|
+ return __ip6_del_rt(rt, NULL);
|
|
}
|
|
}
|
|
|
|
|
|
-static int ip6_route_del(struct in6_rtmsg *rtmsg, struct nlmsghdr *nlh,
|
|
|
|
- void *_rtattr, struct netlink_skb_parms *req,
|
|
|
|
- u32 table_id)
|
|
|
|
|
|
+static int ip6_route_del(struct fib6_config *cfg)
|
|
{
|
|
{
|
|
struct fib6_table *table;
|
|
struct fib6_table *table;
|
|
struct fib6_node *fn;
|
|
struct fib6_node *fn;
|
|
struct rt6_info *rt;
|
|
struct rt6_info *rt;
|
|
int err = -ESRCH;
|
|
int err = -ESRCH;
|
|
|
|
|
|
- table = fib6_get_table(table_id);
|
|
|
|
|
|
+ table = fib6_get_table(cfg->fc_table);
|
|
if (table == NULL)
|
|
if (table == NULL)
|
|
return err;
|
|
return err;
|
|
|
|
|
|
read_lock_bh(&table->tb6_lock);
|
|
read_lock_bh(&table->tb6_lock);
|
|
|
|
|
|
fn = fib6_locate(&table->tb6_root,
|
|
fn = fib6_locate(&table->tb6_root,
|
|
- &rtmsg->rtmsg_dst, rtmsg->rtmsg_dst_len,
|
|
|
|
- &rtmsg->rtmsg_src, rtmsg->rtmsg_src_len);
|
|
|
|
|
|
+ &cfg->fc_dst, cfg->fc_dst_len,
|
|
|
|
+ &cfg->fc_src, cfg->fc_src_len);
|
|
|
|
|
|
if (fn) {
|
|
if (fn) {
|
|
for (rt = fn->leaf; rt; rt = rt->u.next) {
|
|
for (rt = fn->leaf; rt; rt = rt->u.next) {
|
|
- if (rtmsg->rtmsg_ifindex &&
|
|
|
|
|
|
+ if (cfg->fc_ifindex &&
|
|
(rt->rt6i_dev == NULL ||
|
|
(rt->rt6i_dev == NULL ||
|
|
- rt->rt6i_dev->ifindex != rtmsg->rtmsg_ifindex))
|
|
|
|
|
|
+ rt->rt6i_dev->ifindex != cfg->fc_ifindex))
|
|
continue;
|
|
continue;
|
|
- if (rtmsg->rtmsg_flags&RTF_GATEWAY &&
|
|
|
|
- !ipv6_addr_equal(&rtmsg->rtmsg_gateway, &rt->rt6i_gateway))
|
|
|
|
|
|
+ if (cfg->fc_flags & RTF_GATEWAY &&
|
|
|
|
+ !ipv6_addr_equal(&cfg->fc_gateway, &rt->rt6i_gateway))
|
|
continue;
|
|
continue;
|
|
- if (rtmsg->rtmsg_metric &&
|
|
|
|
- rtmsg->rtmsg_metric != rt->rt6i_metric)
|
|
|
|
|
|
+ if (cfg->fc_metric && cfg->fc_metric != rt->rt6i_metric)
|
|
continue;
|
|
continue;
|
|
dst_hold(&rt->u.dst);
|
|
dst_hold(&rt->u.dst);
|
|
read_unlock_bh(&table->tb6_lock);
|
|
read_unlock_bh(&table->tb6_lock);
|
|
|
|
|
|
- return __ip6_del_rt(rt, nlh, _rtattr, req);
|
|
|
|
|
|
+ return __ip6_del_rt(rt, &cfg->fc_nlinfo);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
read_unlock_bh(&table->tb6_lock);
|
|
read_unlock_bh(&table->tb6_lock);
|
|
@@ -1565,21 +1551,23 @@ static struct rt6_info *rt6_add_route_info(struct in6_addr *prefix, int prefixle
|
|
struct in6_addr *gwaddr, int ifindex,
|
|
struct in6_addr *gwaddr, int ifindex,
|
|
unsigned pref)
|
|
unsigned pref)
|
|
{
|
|
{
|
|
- struct in6_rtmsg rtmsg;
|
|
|
|
|
|
+ struct fib6_config cfg = {
|
|
|
|
+ .fc_table = RT6_TABLE_INFO,
|
|
|
|
+ .fc_metric = 1024,
|
|
|
|
+ .fc_ifindex = ifindex,
|
|
|
|
+ .fc_dst_len = prefixlen,
|
|
|
|
+ .fc_flags = RTF_GATEWAY | RTF_ADDRCONF | RTF_ROUTEINFO |
|
|
|
|
+ RTF_UP | RTF_PREF(pref),
|
|
|
|
+ };
|
|
|
|
+
|
|
|
|
+ ipv6_addr_copy(&cfg.fc_dst, prefix);
|
|
|
|
+ ipv6_addr_copy(&cfg.fc_gateway, gwaddr);
|
|
|
|
|
|
- memset(&rtmsg, 0, sizeof(rtmsg));
|
|
|
|
- rtmsg.rtmsg_type = RTMSG_NEWROUTE;
|
|
|
|
- ipv6_addr_copy(&rtmsg.rtmsg_dst, prefix);
|
|
|
|
- rtmsg.rtmsg_dst_len = prefixlen;
|
|
|
|
- ipv6_addr_copy(&rtmsg.rtmsg_gateway, gwaddr);
|
|
|
|
- rtmsg.rtmsg_metric = 1024;
|
|
|
|
- rtmsg.rtmsg_flags = RTF_GATEWAY | RTF_ADDRCONF | RTF_ROUTEINFO | RTF_UP | RTF_PREF(pref);
|
|
|
|
/* We should treat it as a default route if prefix length is 0. */
|
|
/* We should treat it as a default route if prefix length is 0. */
|
|
if (!prefixlen)
|
|
if (!prefixlen)
|
|
- rtmsg.rtmsg_flags |= RTF_DEFAULT;
|
|
|
|
- rtmsg.rtmsg_ifindex = ifindex;
|
|
|
|
|
|
+ cfg.fc_flags |= RTF_DEFAULT;
|
|
|
|
|
|
- ip6_route_add(&rtmsg, NULL, NULL, NULL, RT6_TABLE_INFO);
|
|
|
|
|
|
+ ip6_route_add(&cfg);
|
|
|
|
|
|
return rt6_get_route_info(prefix, prefixlen, gwaddr, ifindex);
|
|
return rt6_get_route_info(prefix, prefixlen, gwaddr, ifindex);
|
|
}
|
|
}
|
|
@@ -1611,18 +1599,18 @@ struct rt6_info *rt6_add_dflt_router(struct in6_addr *gwaddr,
|
|
struct net_device *dev,
|
|
struct net_device *dev,
|
|
unsigned int pref)
|
|
unsigned int pref)
|
|
{
|
|
{
|
|
- struct in6_rtmsg rtmsg;
|
|
|
|
|
|
+ struct fib6_config cfg = {
|
|
|
|
+ .fc_table = RT6_TABLE_DFLT,
|
|
|
|
+ .fc_metric = 1024,
|
|
|
|
+ .fc_ifindex = dev->ifindex,
|
|
|
|
+ .fc_flags = RTF_GATEWAY | RTF_ADDRCONF | RTF_DEFAULT |
|
|
|
|
+ RTF_UP | RTF_EXPIRES | RTF_PREF(pref),
|
|
|
|
+ };
|
|
|
|
|
|
- memset(&rtmsg, 0, sizeof(struct in6_rtmsg));
|
|
|
|
- rtmsg.rtmsg_type = RTMSG_NEWROUTE;
|
|
|
|
- ipv6_addr_copy(&rtmsg.rtmsg_gateway, gwaddr);
|
|
|
|
- rtmsg.rtmsg_metric = 1024;
|
|
|
|
- rtmsg.rtmsg_flags = RTF_GATEWAY | RTF_ADDRCONF | RTF_DEFAULT | RTF_UP | RTF_EXPIRES |
|
|
|
|
- RTF_PREF(pref);
|
|
|
|
|
|
+ ipv6_addr_copy(&cfg.fc_gateway, gwaddr);
|
|
|
|
|
|
- rtmsg.rtmsg_ifindex = dev->ifindex;
|
|
|
|
|
|
+ ip6_route_add(&cfg);
|
|
|
|
|
|
- ip6_route_add(&rtmsg, NULL, NULL, NULL, RT6_TABLE_DFLT);
|
|
|
|
return rt6_get_dflt_router(gwaddr, dev);
|
|
return rt6_get_dflt_router(gwaddr, dev);
|
|
}
|
|
}
|
|
|
|
|
|
@@ -1649,8 +1637,27 @@ restart:
|
|
read_unlock_bh(&table->tb6_lock);
|
|
read_unlock_bh(&table->tb6_lock);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+static void rtmsg_to_fib6_config(struct in6_rtmsg *rtmsg,
|
|
|
|
+ struct fib6_config *cfg)
|
|
|
|
+{
|
|
|
|
+ memset(cfg, 0, sizeof(*cfg));
|
|
|
|
+
|
|
|
|
+ cfg->fc_table = RT6_TABLE_MAIN;
|
|
|
|
+ cfg->fc_ifindex = rtmsg->rtmsg_ifindex;
|
|
|
|
+ cfg->fc_metric = rtmsg->rtmsg_metric;
|
|
|
|
+ cfg->fc_expires = rtmsg->rtmsg_info;
|
|
|
|
+ cfg->fc_dst_len = rtmsg->rtmsg_dst_len;
|
|
|
|
+ cfg->fc_src_len = rtmsg->rtmsg_src_len;
|
|
|
|
+ cfg->fc_flags = rtmsg->rtmsg_flags;
|
|
|
|
+
|
|
|
|
+ ipv6_addr_copy(&cfg->fc_dst, &rtmsg->rtmsg_dst);
|
|
|
|
+ ipv6_addr_copy(&cfg->fc_src, &rtmsg->rtmsg_src);
|
|
|
|
+ ipv6_addr_copy(&cfg->fc_gateway, &rtmsg->rtmsg_gateway);
|
|
|
|
+}
|
|
|
|
+
|
|
int ipv6_route_ioctl(unsigned int cmd, void __user *arg)
|
|
int ipv6_route_ioctl(unsigned int cmd, void __user *arg)
|
|
{
|
|
{
|
|
|
|
+ struct fib6_config cfg;
|
|
struct in6_rtmsg rtmsg;
|
|
struct in6_rtmsg rtmsg;
|
|
int err;
|
|
int err;
|
|
|
|
|
|
@@ -1663,16 +1670,16 @@ int ipv6_route_ioctl(unsigned int cmd, void __user *arg)
|
|
sizeof(struct in6_rtmsg));
|
|
sizeof(struct in6_rtmsg));
|
|
if (err)
|
|
if (err)
|
|
return -EFAULT;
|
|
return -EFAULT;
|
|
-
|
|
|
|
|
|
+
|
|
|
|
+ rtmsg_to_fib6_config(&rtmsg, &cfg);
|
|
|
|
+
|
|
rtnl_lock();
|
|
rtnl_lock();
|
|
switch (cmd) {
|
|
switch (cmd) {
|
|
case SIOCADDRT:
|
|
case SIOCADDRT:
|
|
- err = ip6_route_add(&rtmsg, NULL, NULL, NULL,
|
|
|
|
- RT6_TABLE_MAIN);
|
|
|
|
|
|
+ err = ip6_route_add(&cfg);
|
|
break;
|
|
break;
|
|
case SIOCDELRT:
|
|
case SIOCDELRT:
|
|
- err = ip6_route_del(&rtmsg, NULL, NULL, NULL,
|
|
|
|
- RT6_TABLE_MAIN);
|
|
|
|
|
|
+ err = ip6_route_del(&cfg);
|
|
break;
|
|
break;
|
|
default:
|
|
default:
|
|
err = -EINVAL;
|
|
err = -EINVAL;
|
|
@@ -1823,66 +1830,104 @@ void rt6_mtu_change(struct net_device *dev, unsigned mtu)
|
|
fib6_clean_all(rt6_mtu_change_route, 0, &arg);
|
|
fib6_clean_all(rt6_mtu_change_route, 0, &arg);
|
|
}
|
|
}
|
|
|
|
|
|
-static int inet6_rtm_to_rtmsg(struct rtmsg *r, struct rtattr **rta,
|
|
|
|
- struct in6_rtmsg *rtmsg)
|
|
|
|
|
|
+static struct nla_policy rtm_ipv6_policy[RTA_MAX+1] __read_mostly = {
|
|
|
|
+ [RTA_GATEWAY] = { .minlen = sizeof(struct in6_addr) },
|
|
|
|
+ [RTA_OIF] = { .type = NLA_U32 },
|
|
|
|
+ [RTA_PRIORITY] = { .type = NLA_U32 },
|
|
|
|
+ [RTA_METRICS] = { .type = NLA_NESTED },
|
|
|
|
+};
|
|
|
|
+
|
|
|
|
+static int rtm_to_fib6_config(struct sk_buff *skb, struct nlmsghdr *nlh,
|
|
|
|
+ struct fib6_config *cfg)
|
|
{
|
|
{
|
|
- memset(rtmsg, 0, sizeof(*rtmsg));
|
|
|
|
|
|
+ struct rtmsg *rtm;
|
|
|
|
+ struct nlattr *tb[RTA_MAX+1];
|
|
|
|
+ int err;
|
|
|
|
+
|
|
|
|
+ err = nlmsg_parse(nlh, sizeof(*rtm), tb, RTA_MAX, rtm_ipv6_policy);
|
|
|
|
+ if (err < 0)
|
|
|
|
+ goto errout;
|
|
|
|
|
|
- rtmsg->rtmsg_dst_len = r->rtm_dst_len;
|
|
|
|
- rtmsg->rtmsg_src_len = r->rtm_src_len;
|
|
|
|
- rtmsg->rtmsg_flags = RTF_UP;
|
|
|
|
- if (r->rtm_type == RTN_UNREACHABLE)
|
|
|
|
- rtmsg->rtmsg_flags |= RTF_REJECT;
|
|
|
|
|
|
+ err = -EINVAL;
|
|
|
|
+ rtm = nlmsg_data(nlh);
|
|
|
|
+ memset(cfg, 0, sizeof(*cfg));
|
|
|
|
|
|
- if (rta[RTA_GATEWAY-1]) {
|
|
|
|
- if (rta[RTA_GATEWAY-1]->rta_len != RTA_LENGTH(16))
|
|
|
|
- return -EINVAL;
|
|
|
|
- memcpy(&rtmsg->rtmsg_gateway, RTA_DATA(rta[RTA_GATEWAY-1]), 16);
|
|
|
|
- rtmsg->rtmsg_flags |= RTF_GATEWAY;
|
|
|
|
- }
|
|
|
|
- if (rta[RTA_DST-1]) {
|
|
|
|
- if (RTA_PAYLOAD(rta[RTA_DST-1]) < ((r->rtm_dst_len+7)>>3))
|
|
|
|
- return -EINVAL;
|
|
|
|
- memcpy(&rtmsg->rtmsg_dst, RTA_DATA(rta[RTA_DST-1]), ((r->rtm_dst_len+7)>>3));
|
|
|
|
|
|
+ cfg->fc_table = rtm->rtm_table;
|
|
|
|
+ cfg->fc_dst_len = rtm->rtm_dst_len;
|
|
|
|
+ cfg->fc_src_len = rtm->rtm_src_len;
|
|
|
|
+ cfg->fc_flags = RTF_UP;
|
|
|
|
+ cfg->fc_protocol = rtm->rtm_protocol;
|
|
|
|
+
|
|
|
|
+ if (rtm->rtm_type == RTN_UNREACHABLE)
|
|
|
|
+ cfg->fc_flags |= RTF_REJECT;
|
|
|
|
+
|
|
|
|
+ cfg->fc_nlinfo.pid = NETLINK_CB(skb).pid;
|
|
|
|
+ cfg->fc_nlinfo.nlh = nlh;
|
|
|
|
+
|
|
|
|
+ if (tb[RTA_GATEWAY]) {
|
|
|
|
+ nla_memcpy(&cfg->fc_gateway, tb[RTA_GATEWAY], 16);
|
|
|
|
+ cfg->fc_flags |= RTF_GATEWAY;
|
|
}
|
|
}
|
|
- if (rta[RTA_SRC-1]) {
|
|
|
|
- if (RTA_PAYLOAD(rta[RTA_SRC-1]) < ((r->rtm_src_len+7)>>3))
|
|
|
|
- return -EINVAL;
|
|
|
|
- memcpy(&rtmsg->rtmsg_src, RTA_DATA(rta[RTA_SRC-1]), ((r->rtm_src_len+7)>>3));
|
|
|
|
|
|
+
|
|
|
|
+ if (tb[RTA_DST]) {
|
|
|
|
+ int plen = (rtm->rtm_dst_len + 7) >> 3;
|
|
|
|
+
|
|
|
|
+ if (nla_len(tb[RTA_DST]) < plen)
|
|
|
|
+ goto errout;
|
|
|
|
+
|
|
|
|
+ nla_memcpy(&cfg->fc_dst, tb[RTA_DST], plen);
|
|
}
|
|
}
|
|
- if (rta[RTA_OIF-1]) {
|
|
|
|
- if (rta[RTA_OIF-1]->rta_len != RTA_LENGTH(sizeof(int)))
|
|
|
|
- return -EINVAL;
|
|
|
|
- memcpy(&rtmsg->rtmsg_ifindex, RTA_DATA(rta[RTA_OIF-1]), sizeof(int));
|
|
|
|
|
|
+
|
|
|
|
+ if (tb[RTA_SRC]) {
|
|
|
|
+ int plen = (rtm->rtm_src_len + 7) >> 3;
|
|
|
|
+
|
|
|
|
+ if (nla_len(tb[RTA_SRC]) < plen)
|
|
|
|
+ goto errout;
|
|
|
|
+
|
|
|
|
+ nla_memcpy(&cfg->fc_src, tb[RTA_SRC], plen);
|
|
}
|
|
}
|
|
- if (rta[RTA_PRIORITY-1]) {
|
|
|
|
- if (rta[RTA_PRIORITY-1]->rta_len != RTA_LENGTH(4))
|
|
|
|
- return -EINVAL;
|
|
|
|
- memcpy(&rtmsg->rtmsg_metric, RTA_DATA(rta[RTA_PRIORITY-1]), 4);
|
|
|
|
|
|
+
|
|
|
|
+ if (tb[RTA_OIF])
|
|
|
|
+ cfg->fc_ifindex = nla_get_u32(tb[RTA_OIF]);
|
|
|
|
+
|
|
|
|
+ if (tb[RTA_PRIORITY])
|
|
|
|
+ cfg->fc_metric = nla_get_u32(tb[RTA_PRIORITY]);
|
|
|
|
+
|
|
|
|
+ if (tb[RTA_METRICS]) {
|
|
|
|
+ cfg->fc_mx = nla_data(tb[RTA_METRICS]);
|
|
|
|
+ cfg->fc_mx_len = nla_len(tb[RTA_METRICS]);
|
|
}
|
|
}
|
|
- return 0;
|
|
|
|
|
|
+
|
|
|
|
+ if (tb[RTA_TABLE])
|
|
|
|
+ cfg->fc_table = nla_get_u32(tb[RTA_TABLE]);
|
|
|
|
+
|
|
|
|
+ err = 0;
|
|
|
|
+errout:
|
|
|
|
+ return err;
|
|
}
|
|
}
|
|
|
|
|
|
int inet6_rtm_delroute(struct sk_buff *skb, struct nlmsghdr* nlh, void *arg)
|
|
int inet6_rtm_delroute(struct sk_buff *skb, struct nlmsghdr* nlh, void *arg)
|
|
{
|
|
{
|
|
- struct rtmsg *r = NLMSG_DATA(nlh);
|
|
|
|
- struct in6_rtmsg rtmsg;
|
|
|
|
|
|
+ struct fib6_config cfg;
|
|
|
|
+ int err;
|
|
|
|
|
|
- if (inet6_rtm_to_rtmsg(r, arg, &rtmsg))
|
|
|
|
- return -EINVAL;
|
|
|
|
- return ip6_route_del(&rtmsg, nlh, arg, &NETLINK_CB(skb),
|
|
|
|
- rtm_get_table(arg, r->rtm_table));
|
|
|
|
|
|
+ err = rtm_to_fib6_config(skb, nlh, &cfg);
|
|
|
|
+ if (err < 0)
|
|
|
|
+ return err;
|
|
|
|
+
|
|
|
|
+ return ip6_route_del(&cfg);
|
|
}
|
|
}
|
|
|
|
|
|
int inet6_rtm_newroute(struct sk_buff *skb, struct nlmsghdr* nlh, void *arg)
|
|
int inet6_rtm_newroute(struct sk_buff *skb, struct nlmsghdr* nlh, void *arg)
|
|
{
|
|
{
|
|
- struct rtmsg *r = NLMSG_DATA(nlh);
|
|
|
|
- struct in6_rtmsg rtmsg;
|
|
|
|
|
|
+ struct fib6_config cfg;
|
|
|
|
+ int err;
|
|
|
|
|
|
- if (inet6_rtm_to_rtmsg(r, arg, &rtmsg))
|
|
|
|
- return -EINVAL;
|
|
|
|
- return ip6_route_add(&rtmsg, nlh, arg, &NETLINK_CB(skb),
|
|
|
|
- rtm_get_table(arg, r->rtm_table));
|
|
|
|
|
|
+ err = rtm_to_fib6_config(skb, nlh, &cfg);
|
|
|
|
+ if (err < 0)
|
|
|
|
+ return err;
|
|
|
|
+
|
|
|
|
+ return ip6_route_add(&cfg);
|
|
}
|
|
}
|
|
|
|
|
|
static int rt6_fill_node(struct sk_buff *skb, struct rt6_info *rt,
|
|
static int rt6_fill_node(struct sk_buff *skb, struct rt6_info *rt,
|
|
@@ -2063,15 +2108,21 @@ out_free:
|
|
goto out;
|
|
goto out;
|
|
}
|
|
}
|
|
|
|
|
|
-void inet6_rt_notify(int event, struct rt6_info *rt, struct nlmsghdr *nlh,
|
|
|
|
- struct netlink_skb_parms *req)
|
|
|
|
|
|
+void inet6_rt_notify(int event, struct rt6_info *rt, struct nl_info *info)
|
|
{
|
|
{
|
|
struct sk_buff *skb;
|
|
struct sk_buff *skb;
|
|
- u32 pid = req ? req->pid : 0;
|
|
|
|
- u32 seq = nlh ? nlh->nlmsg_seq : 0;
|
|
|
|
|
|
+ u32 pid = 0, seq = 0;
|
|
|
|
+ struct nlmsghdr *nlh = NULL;
|
|
int payload = sizeof(struct rtmsg) + 256;
|
|
int payload = sizeof(struct rtmsg) + 256;
|
|
int err = -ENOBUFS;
|
|
int err = -ENOBUFS;
|
|
|
|
|
|
|
|
+ if (info) {
|
|
|
|
+ pid = info->pid;
|
|
|
|
+ nlh = info->nlh;
|
|
|
|
+ if (nlh)
|
|
|
|
+ seq = nlh->nlmsg_seq;
|
|
|
|
+ }
|
|
|
|
+
|
|
skb = nlmsg_new(nlmsg_total_size(payload), gfp_any());
|
|
skb = nlmsg_new(nlmsg_total_size(payload), gfp_any());
|
|
if (skb == NULL)
|
|
if (skb == NULL)
|
|
goto errout;
|
|
goto errout;
|