diff options
-rw-r--r-- | net/core/neighbour.c | 90 |
1 files changed, 51 insertions, 39 deletions
diff --git a/net/core/neighbour.c b/net/core/neighbour.c index 39c07cc66ee7..6036f43c1fd6 100644 --- a/net/core/neighbour.c +++ b/net/core/neighbour.c | |||
@@ -1506,76 +1506,88 @@ out: | |||
1506 | 1506 | ||
1507 | int neigh_add(struct sk_buff *skb, struct nlmsghdr *nlh, void *arg) | 1507 | int neigh_add(struct sk_buff *skb, struct nlmsghdr *nlh, void *arg) |
1508 | { | 1508 | { |
1509 | struct ndmsg *ndm = NLMSG_DATA(nlh); | 1509 | struct ndmsg *ndm; |
1510 | struct rtattr **nda = arg; | 1510 | struct nlattr *tb[NDA_MAX+1]; |
1511 | struct neigh_table *tbl; | 1511 | struct neigh_table *tbl; |
1512 | struct net_device *dev = NULL; | 1512 | struct net_device *dev = NULL; |
1513 | int err = -ENODEV; | 1513 | int err; |
1514 | |||
1515 | err = nlmsg_parse(nlh, sizeof(*ndm), tb, NDA_MAX, NULL); | ||
1516 | if (err < 0) | ||
1517 | goto out; | ||
1514 | 1518 | ||
1515 | if (ndm->ndm_ifindex && | 1519 | err = -EINVAL; |
1516 | (dev = dev_get_by_index(ndm->ndm_ifindex)) == NULL) | 1520 | if (tb[NDA_DST] == NULL) |
1517 | goto out; | 1521 | goto out; |
1518 | 1522 | ||
1523 | ndm = nlmsg_data(nlh); | ||
1524 | if (ndm->ndm_ifindex) { | ||
1525 | dev = dev_get_by_index(ndm->ndm_ifindex); | ||
1526 | if (dev == NULL) { | ||
1527 | err = -ENODEV; | ||
1528 | goto out; | ||
1529 | } | ||
1530 | |||
1531 | if (tb[NDA_LLADDR] && nla_len(tb[NDA_LLADDR]) < dev->addr_len) | ||
1532 | goto out_dev_put; | ||
1533 | } | ||
1534 | |||
1519 | read_lock(&neigh_tbl_lock); | 1535 | read_lock(&neigh_tbl_lock); |
1520 | for (tbl = neigh_tables; tbl; tbl = tbl->next) { | 1536 | for (tbl = neigh_tables; tbl; tbl = tbl->next) { |
1521 | struct rtattr *lladdr_attr = nda[NDA_LLADDR - 1]; | 1537 | int flags = NEIGH_UPDATE_F_ADMIN | NEIGH_UPDATE_F_OVERRIDE; |
1522 | struct rtattr *dst_attr = nda[NDA_DST - 1]; | 1538 | struct neighbour *neigh; |
1523 | int override = 1; | 1539 | void *dst, *lladdr; |
1524 | struct neighbour *n; | ||
1525 | 1540 | ||
1526 | if (tbl->family != ndm->ndm_family) | 1541 | if (tbl->family != ndm->ndm_family) |
1527 | continue; | 1542 | continue; |
1528 | read_unlock(&neigh_tbl_lock); | 1543 | read_unlock(&neigh_tbl_lock); |
1529 | 1544 | ||
1530 | err = -EINVAL; | 1545 | if (nla_len(tb[NDA_DST]) < tbl->key_len) |
1531 | if (!dst_attr || RTA_PAYLOAD(dst_attr) < tbl->key_len) | ||
1532 | goto out_dev_put; | 1546 | goto out_dev_put; |
1547 | dst = nla_data(tb[NDA_DST]); | ||
1548 | lladdr = tb[NDA_LLADDR] ? nla_data(tb[NDA_LLADDR]) : NULL; | ||
1533 | 1549 | ||
1534 | if (ndm->ndm_flags & NTF_PROXY) { | 1550 | if (ndm->ndm_flags & NTF_PROXY) { |
1535 | err = -ENOBUFS; | 1551 | err = 0; |
1536 | if (pneigh_lookup(tbl, RTA_DATA(dst_attr), dev, 1)) | 1552 | if (pneigh_lookup(tbl, dst, dev, 1) == NULL) |
1537 | err = 0; | 1553 | err = -ENOBUFS; |
1538 | goto out_dev_put; | 1554 | goto out_dev_put; |
1539 | } | 1555 | } |
1540 | 1556 | ||
1541 | err = -EINVAL; | 1557 | if (dev == NULL) |
1542 | if (!dev) | ||
1543 | goto out; | ||
1544 | if (lladdr_attr && RTA_PAYLOAD(lladdr_attr) < dev->addr_len) | ||
1545 | goto out_dev_put; | 1558 | goto out_dev_put; |
1559 | |||
1560 | neigh = neigh_lookup(tbl, dst, dev); | ||
1561 | if (neigh == NULL) { | ||
1562 | if (!(nlh->nlmsg_flags & NLM_F_CREATE)) { | ||
1563 | err = -ENOENT; | ||
1564 | goto out_dev_put; | ||
1565 | } | ||
1546 | 1566 | ||
1547 | n = neigh_lookup(tbl, RTA_DATA(dst_attr), dev); | 1567 | neigh = __neigh_lookup_errno(tbl, dst, dev); |
1548 | if (n) { | 1568 | if (IS_ERR(neigh)) { |
1549 | if (nlh->nlmsg_flags & NLM_F_EXCL) { | 1569 | err = PTR_ERR(neigh); |
1550 | err = -EEXIST; | ||
1551 | neigh_release(n); | ||
1552 | goto out_dev_put; | 1570 | goto out_dev_put; |
1553 | } | 1571 | } |
1554 | |||
1555 | override = nlh->nlmsg_flags & NLM_F_REPLACE; | ||
1556 | } else if (!(nlh->nlmsg_flags & NLM_F_CREATE)) { | ||
1557 | err = -ENOENT; | ||
1558 | goto out_dev_put; | ||
1559 | } else { | 1572 | } else { |
1560 | n = __neigh_lookup_errno(tbl, RTA_DATA(dst_attr), dev); | 1573 | if (nlh->nlmsg_flags & NLM_F_EXCL) { |
1561 | if (IS_ERR(n)) { | 1574 | err = -EEXIST; |
1562 | err = PTR_ERR(n); | 1575 | neigh_release(neigh); |
1563 | goto out_dev_put; | 1576 | goto out_dev_put; |
1564 | } | 1577 | } |
1565 | } | ||
1566 | 1578 | ||
1567 | err = neigh_update(n, | 1579 | if (!(nlh->nlmsg_flags & NLM_F_REPLACE)) |
1568 | lladdr_attr ? RTA_DATA(lladdr_attr) : NULL, | 1580 | flags &= ~NEIGH_UPDATE_F_OVERRIDE; |
1569 | ndm->ndm_state, | 1581 | } |
1570 | (override ? NEIGH_UPDATE_F_OVERRIDE : 0) | | ||
1571 | NEIGH_UPDATE_F_ADMIN); | ||
1572 | 1582 | ||
1573 | neigh_release(n); | 1583 | err = neigh_update(neigh, lladdr, ndm->ndm_state, flags); |
1584 | neigh_release(neigh); | ||
1574 | goto out_dev_put; | 1585 | goto out_dev_put; |
1575 | } | 1586 | } |
1576 | 1587 | ||
1577 | read_unlock(&neigh_tbl_lock); | 1588 | read_unlock(&neigh_tbl_lock); |
1578 | err = -EADDRNOTAVAIL; | 1589 | err = -EAFNOSUPPORT; |
1590 | |||
1579 | out_dev_put: | 1591 | out_dev_put: |
1580 | if (dev) | 1592 | if (dev) |
1581 | dev_put(dev); | 1593 | dev_put(dev); |