diff options
Diffstat (limited to 'net/ipv4/route.c')
-rw-r--r-- | net/ipv4/route.c | 84 |
1 files changed, 45 insertions, 39 deletions
diff --git a/net/ipv4/route.c b/net/ipv4/route.c index 31b67059ac29..a4d4cb85a16c 100644 --- a/net/ipv4/route.c +++ b/net/ipv4/route.c | |||
@@ -2737,18 +2737,24 @@ nla_put_failure: | |||
2737 | 2737 | ||
2738 | int inet_rtm_getroute(struct sk_buff *in_skb, struct nlmsghdr* nlh, void *arg) | 2738 | int inet_rtm_getroute(struct sk_buff *in_skb, struct nlmsghdr* nlh, void *arg) |
2739 | { | 2739 | { |
2740 | struct rtattr **rta = arg; | 2740 | struct rtmsg *rtm; |
2741 | struct rtmsg *rtm = NLMSG_DATA(nlh); | 2741 | struct nlattr *tb[RTA_MAX+1]; |
2742 | struct rtable *rt = NULL; | 2742 | struct rtable *rt = NULL; |
2743 | u32 dst = 0; | 2743 | u32 dst, src, iif; |
2744 | u32 src = 0; | 2744 | int err; |
2745 | int iif = 0; | ||
2746 | int err = -ENOBUFS; | ||
2747 | struct sk_buff *skb; | 2745 | struct sk_buff *skb; |
2748 | 2746 | ||
2747 | err = nlmsg_parse(nlh, sizeof(*rtm), tb, RTA_MAX, rtm_ipv4_policy); | ||
2748 | if (err < 0) | ||
2749 | goto errout; | ||
2750 | |||
2751 | rtm = nlmsg_data(nlh); | ||
2752 | |||
2749 | skb = alloc_skb(NLMSG_GOODSIZE, GFP_KERNEL); | 2753 | skb = alloc_skb(NLMSG_GOODSIZE, GFP_KERNEL); |
2750 | if (!skb) | 2754 | if (skb == NULL) { |
2751 | goto out; | 2755 | err = -ENOBUFS; |
2756 | goto errout; | ||
2757 | } | ||
2752 | 2758 | ||
2753 | /* Reserve room for dummy headers, this skb can pass | 2759 | /* Reserve room for dummy headers, this skb can pass |
2754 | through good chunk of routing engine. | 2760 | through good chunk of routing engine. |
@@ -2759,61 +2765,61 @@ int inet_rtm_getroute(struct sk_buff *in_skb, struct nlmsghdr* nlh, void *arg) | |||
2759 | skb->nh.iph->protocol = IPPROTO_ICMP; | 2765 | skb->nh.iph->protocol = IPPROTO_ICMP; |
2760 | skb_reserve(skb, MAX_HEADER + sizeof(struct iphdr)); | 2766 | skb_reserve(skb, MAX_HEADER + sizeof(struct iphdr)); |
2761 | 2767 | ||
2762 | if (rta[RTA_SRC - 1]) | 2768 | src = tb[RTA_SRC] ? nla_get_u32(tb[RTA_SRC]) : 0; |
2763 | memcpy(&src, RTA_DATA(rta[RTA_SRC - 1]), 4); | 2769 | dst = tb[RTA_DST] ? nla_get_u32(tb[RTA_DST]) : 0; |
2764 | if (rta[RTA_DST - 1]) | 2770 | iif = tb[RTA_IIF] ? nla_get_u32(tb[RTA_IIF]) : 0; |
2765 | memcpy(&dst, RTA_DATA(rta[RTA_DST - 1]), 4); | ||
2766 | if (rta[RTA_IIF - 1]) | ||
2767 | memcpy(&iif, RTA_DATA(rta[RTA_IIF - 1]), sizeof(int)); | ||
2768 | 2771 | ||
2769 | if (iif) { | 2772 | if (iif) { |
2770 | struct net_device *dev = __dev_get_by_index(iif); | 2773 | struct net_device *dev; |
2771 | err = -ENODEV; | 2774 | |
2772 | if (!dev) | 2775 | dev = __dev_get_by_index(iif); |
2773 | goto out_free; | 2776 | if (dev == NULL) { |
2777 | err = -ENODEV; | ||
2778 | goto errout_free; | ||
2779 | } | ||
2780 | |||
2774 | skb->protocol = htons(ETH_P_IP); | 2781 | skb->protocol = htons(ETH_P_IP); |
2775 | skb->dev = dev; | 2782 | skb->dev = dev; |
2776 | local_bh_disable(); | 2783 | local_bh_disable(); |
2777 | err = ip_route_input(skb, dst, src, rtm->rtm_tos, dev); | 2784 | err = ip_route_input(skb, dst, src, rtm->rtm_tos, dev); |
2778 | local_bh_enable(); | 2785 | local_bh_enable(); |
2779 | rt = (struct rtable*)skb->dst; | 2786 | |
2780 | if (!err && rt->u.dst.error) | 2787 | rt = (struct rtable*) skb->dst; |
2788 | if (err == 0 && rt->u.dst.error) | ||
2781 | err = -rt->u.dst.error; | 2789 | err = -rt->u.dst.error; |
2782 | } else { | 2790 | } else { |
2783 | struct flowi fl = { .nl_u = { .ip4_u = { .daddr = dst, | 2791 | struct flowi fl = { |
2784 | .saddr = src, | 2792 | .nl_u = { |
2785 | .tos = rtm->rtm_tos } } }; | 2793 | .ip4_u = { |
2786 | int oif = 0; | 2794 | .daddr = dst, |
2787 | if (rta[RTA_OIF - 1]) | 2795 | .saddr = src, |
2788 | memcpy(&oif, RTA_DATA(rta[RTA_OIF - 1]), sizeof(int)); | 2796 | .tos = rtm->rtm_tos, |
2789 | fl.oif = oif; | 2797 | }, |
2798 | }, | ||
2799 | .oif = tb[RTA_OIF] ? nla_get_u32(tb[RTA_OIF]) : 0, | ||
2800 | }; | ||
2790 | err = ip_route_output_key(&rt, &fl); | 2801 | err = ip_route_output_key(&rt, &fl); |
2791 | } | 2802 | } |
2803 | |||
2792 | if (err) | 2804 | if (err) |
2793 | goto out_free; | 2805 | goto errout_free; |
2794 | 2806 | ||
2795 | skb->dst = &rt->u.dst; | 2807 | skb->dst = &rt->u.dst; |
2796 | if (rtm->rtm_flags & RTM_F_NOTIFY) | 2808 | if (rtm->rtm_flags & RTM_F_NOTIFY) |
2797 | rt->rt_flags |= RTCF_NOTIFY; | 2809 | rt->rt_flags |= RTCF_NOTIFY; |
2798 | 2810 | ||
2799 | NETLINK_CB(skb).dst_pid = NETLINK_CB(in_skb).pid; | ||
2800 | |||
2801 | err = rt_fill_info(skb, NETLINK_CB(in_skb).pid, nlh->nlmsg_seq, | 2811 | err = rt_fill_info(skb, NETLINK_CB(in_skb).pid, nlh->nlmsg_seq, |
2802 | RTM_NEWROUTE, 0, 0); | 2812 | RTM_NEWROUTE, 0, 0); |
2803 | if (!err) | 2813 | if (err <= 0) |
2804 | goto out_free; | 2814 | goto errout_free; |
2805 | if (err < 0) { | ||
2806 | err = -EMSGSIZE; | ||
2807 | goto out_free; | ||
2808 | } | ||
2809 | 2815 | ||
2810 | err = rtnl_unicast(skb, NETLINK_CB(in_skb).pid); | 2816 | err = rtnl_unicast(skb, NETLINK_CB(in_skb).pid); |
2811 | out: | 2817 | errout: |
2812 | return err; | 2818 | return err; |
2813 | 2819 | ||
2814 | out_free: | 2820 | errout_free: |
2815 | kfree_skb(skb); | 2821 | kfree_skb(skb); |
2816 | goto out; | 2822 | goto errout; |
2817 | } | 2823 | } |
2818 | 2824 | ||
2819 | int ip_rt_dump(struct sk_buff *skb, struct netlink_callback *cb) | 2825 | int ip_rt_dump(struct sk_buff *skb, struct netlink_callback *cb) |