aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--net/core/neighbour.c106
1 files changed, 51 insertions, 55 deletions
diff --git a/net/core/neighbour.c b/net/core/neighbour.c
index 6036f43c1fd6..5490afd23b82 100644
--- a/net/core/neighbour.c
+++ b/net/core/neighbour.c
@@ -1901,48 +1901,49 @@ out:
1901 return skb->len; 1901 return skb->len;
1902} 1902}
1903 1903
1904static int neigh_fill_info(struct sk_buff *skb, struct neighbour *n, 1904static int neigh_fill_info(struct sk_buff *skb, struct neighbour *neigh,
1905 u32 pid, u32 seq, int event, unsigned int flags) 1905 u32 pid, u32 seq, int type, unsigned int flags)
1906{ 1906{
1907 unsigned long now = jiffies; 1907 unsigned long now = jiffies;
1908 unsigned char *b = skb->tail;
1909 struct nda_cacheinfo ci; 1908 struct nda_cacheinfo ci;
1910 int locked = 0; 1909 struct nlmsghdr *nlh;
1911 u32 probes; 1910 struct ndmsg *ndm;
1912 struct nlmsghdr *nlh = NLMSG_NEW(skb, pid, seq, event, 1911
1913 sizeof(struct ndmsg), flags); 1912 nlh = nlmsg_put(skb, pid, seq, type, sizeof(*ndm), flags);
1914 struct ndmsg *ndm = NLMSG_DATA(nlh); 1913 if (nlh == NULL)
1914 return -ENOBUFS;
1915 1915
1916 ndm->ndm_family = n->ops->family; 1916 ndm = nlmsg_data(nlh);
1917 ndm->ndm_family = neigh->ops->family;
1917 ndm->ndm_pad1 = 0; 1918 ndm->ndm_pad1 = 0;
1918 ndm->ndm_pad2 = 0; 1919 ndm->ndm_pad2 = 0;
1919 ndm->ndm_flags = n->flags; 1920 ndm->ndm_flags = neigh->flags;
1920 ndm->ndm_type = n->type; 1921 ndm->ndm_type = neigh->type;
1921 ndm->ndm_ifindex = n->dev->ifindex; 1922 ndm->ndm_ifindex = neigh->dev->ifindex;
1922 RTA_PUT(skb, NDA_DST, n->tbl->key_len, n->primary_key);
1923 read_lock_bh(&n->lock);
1924 locked = 1;
1925 ndm->ndm_state = n->nud_state;
1926 if (n->nud_state & NUD_VALID)
1927 RTA_PUT(skb, NDA_LLADDR, n->dev->addr_len, n->ha);
1928 ci.ndm_used = now - n->used;
1929 ci.ndm_confirmed = now - n->confirmed;
1930 ci.ndm_updated = now - n->updated;
1931 ci.ndm_refcnt = atomic_read(&n->refcnt) - 1;
1932 probes = atomic_read(&n->probes);
1933 read_unlock_bh(&n->lock);
1934 locked = 0;
1935 RTA_PUT(skb, NDA_CACHEINFO, sizeof(ci), &ci);
1936 RTA_PUT(skb, NDA_PROBES, sizeof(probes), &probes);
1937 nlh->nlmsg_len = skb->tail - b;
1938 return skb->len;
1939 1923
1940nlmsg_failure: 1924 NLA_PUT(skb, NDA_DST, neigh->tbl->key_len, neigh->primary_key);
1941rtattr_failure: 1925
1942 if (locked) 1926 read_lock_bh(&neigh->lock);
1943 read_unlock_bh(&n->lock); 1927 ndm->ndm_state = neigh->nud_state;
1944 skb_trim(skb, b - skb->data); 1928 if ((neigh->nud_state & NUD_VALID) &&
1945 return -1; 1929 nla_put(skb, NDA_LLADDR, neigh->dev->addr_len, neigh->ha) < 0) {
1930 read_unlock_bh(&neigh->lock);
1931 goto nla_put_failure;
1932 }
1933
1934 ci.ndm_used = now - neigh->used;
1935 ci.ndm_confirmed = now - neigh->confirmed;
1936 ci.ndm_updated = now - neigh->updated;
1937 ci.ndm_refcnt = atomic_read(&neigh->refcnt) - 1;
1938 read_unlock_bh(&neigh->lock);
1939
1940 NLA_PUT_U32(skb, NDA_PROBES, atomic_read(&neigh->probes));
1941 NLA_PUT(skb, NDA_CACHEINFO, sizeof(ci), &ci);
1942
1943 return nlmsg_end(skb, nlh);
1944
1945nla_put_failure:
1946 return nlmsg_cancel(skb, nlh);
1946} 1947}
1947 1948
1948 1949
@@ -1986,7 +1987,7 @@ int neigh_dump_info(struct sk_buff *skb, struct netlink_callback *cb)
1986 int t, family, s_t; 1987 int t, family, s_t;
1987 1988
1988 read_lock(&neigh_tbl_lock); 1989 read_lock(&neigh_tbl_lock);
1989 family = ((struct rtgenmsg *)NLMSG_DATA(cb->nlh))->rtgen_family; 1990 family = ((struct rtgenmsg *) nlmsg_data(cb->nlh))->rtgen_family;
1990 s_t = cb->args[0]; 1991 s_t = cb->args[0];
1991 1992
1992 for (tbl = neigh_tables, t = 0; tbl; tbl = tbl->next, t++) { 1993 for (tbl = neigh_tables, t = 0; tbl; tbl = tbl->next, t++) {
@@ -2367,39 +2368,34 @@ static struct file_operations neigh_stat_seq_fops = {
2367#ifdef CONFIG_ARPD 2368#ifdef CONFIG_ARPD
2368void neigh_app_ns(struct neighbour *n) 2369void neigh_app_ns(struct neighbour *n)
2369{ 2370{
2370 struct nlmsghdr *nlh; 2371 struct sk_buff *skb;
2371 int size = NLMSG_SPACE(sizeof(struct ndmsg) + 256);
2372 struct sk_buff *skb = alloc_skb(size, GFP_ATOMIC);
2373 2372
2374 if (!skb) 2373 skb = nlmsg_new(NLMSG_GOODSIZE, GFP_ATOMIC);
2374 if (skb == NULL)
2375 return; 2375 return;
2376 2376
2377 if (neigh_fill_info(skb, n, 0, 0, RTM_GETNEIGH, 0) < 0) { 2377 if (neigh_fill_info(skb, n, 0, 0, RTM_GETNEIGH, NLM_F_REQUEST) <= 0)
2378 kfree_skb(skb); 2378 kfree_skb(skb);
2379 return; 2379 else {
2380 NETLINK_CB(skb).dst_group = RTNLGRP_NEIGH;
2381 netlink_broadcast(rtnl, skb, 0, RTNLGRP_NEIGH, GFP_ATOMIC);
2380 } 2382 }
2381 nlh = (struct nlmsghdr *)skb->data;
2382 nlh->nlmsg_flags = NLM_F_REQUEST;
2383 NETLINK_CB(skb).dst_group = RTNLGRP_NEIGH;
2384 netlink_broadcast(rtnl, skb, 0, RTNLGRP_NEIGH, GFP_ATOMIC);
2385} 2383}
2386 2384
2387static void neigh_app_notify(struct neighbour *n) 2385static void neigh_app_notify(struct neighbour *n)
2388{ 2386{
2389 struct nlmsghdr *nlh; 2387 struct sk_buff *skb;
2390 int size = NLMSG_SPACE(sizeof(struct ndmsg) + 256);
2391 struct sk_buff *skb = alloc_skb(size, GFP_ATOMIC);
2392 2388
2393 if (!skb) 2389 skb = nlmsg_new(NLMSG_GOODSIZE, GFP_ATOMIC);
2390 if (skb == NULL)
2394 return; 2391 return;
2395 2392
2396 if (neigh_fill_info(skb, n, 0, 0, RTM_NEWNEIGH, 0) < 0) { 2393 if (neigh_fill_info(skb, n, 0, 0, RTM_NEWNEIGH, 0) <= 0)
2397 kfree_skb(skb); 2394 kfree_skb(skb);
2398 return; 2395 else {
2396 NETLINK_CB(skb).dst_group = RTNLGRP_NEIGH;
2397 netlink_broadcast(rtnl, skb, 0, RTNLGRP_NEIGH, GFP_ATOMIC);
2399 } 2398 }
2400 nlh = (struct nlmsghdr *)skb->data;
2401 NETLINK_CB(skb).dst_group = RTNLGRP_NEIGH;
2402 netlink_broadcast(rtnl, skb, 0, RTNLGRP_NEIGH, GFP_ATOMIC);
2403} 2399}
2404 2400
2405#endif /* CONFIG_ARPD */ 2401#endif /* CONFIG_ARPD */