Blame view
net/ipv4/datagram.c
2.03 KB
1da177e4c
|
1 2 3 4 5 6 7 8 9 10 11 12 |
/* * common UDP/RAW code * Linux INET implementation * * Authors: * Hideaki YOSHIFUJI <yoshfuji@linux-ipv6.org> * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version * 2 of the License, or (at your option) any later version. */ |
1da177e4c
|
13 14 15 16 |
#include <linux/types.h> #include <linux/module.h> #include <linux/ip.h> #include <linux/in.h> |
20380731b
|
17 |
#include <net/ip.h> |
1da177e4c
|
18 |
#include <net/sock.h> |
1da177e4c
|
19 |
#include <net/route.h> |
c752f0739
|
20 |
#include <net/tcp_states.h> |
1da177e4c
|
21 22 23 24 25 |
int ip4_datagram_connect(struct sock *sk, struct sockaddr *uaddr, int addr_len) { struct inet_sock *inet = inet_sk(sk); struct sockaddr_in *usin = (struct sockaddr_in *) uaddr; |
3038eeac0
|
26 |
struct flowi4 *fl4; |
1da177e4c
|
27 |
struct rtable *rt; |
bada8adc4
|
28 |
__be32 saddr; |
1da177e4c
|
29 30 |
int oif; int err; |
1da177e4c
|
31 |
|
e905a9eda
|
32 33 34 35 36 |
if (addr_len < sizeof(*usin)) return -EINVAL; if (usin->sin_family != AF_INET) return -EAFNOSUPPORT; |
1da177e4c
|
37 38 |
sk_dst_reset(sk); |
3038eeac0
|
39 |
lock_sock(sk); |
1da177e4c
|
40 |
oif = sk->sk_bound_dev_if; |
c720c7e83
|
41 |
saddr = inet->inet_saddr; |
f97c1e0c6
|
42 |
if (ipv4_is_multicast(usin->sin_addr.s_addr)) { |
1da177e4c
|
43 44 45 46 47 |
if (!oif) oif = inet->mc_index; if (!saddr) saddr = inet->mc_addr; } |
3038eeac0
|
48 49 |
fl4 = &inet->cork.fl.u.ip4; rt = ip_route_connect(fl4, usin->sin_addr.s_addr, saddr, |
b23dd4fe4
|
50 51 52 53 54 |
RT_CONN_FLAGS(sk), oif, sk->sk_protocol, inet->inet_sport, usin->sin_port, sk, true); if (IS_ERR(rt)) { err = PTR_ERR(rt); |
584bdf8cb
|
55 |
if (err == -ENETUNREACH) |
7c73a6faf
|
56 |
IP_INC_STATS_BH(sock_net(sk), IPSTATS_MIB_OUTNOROUTES); |
3038eeac0
|
57 |
goto out; |
584bdf8cb
|
58 |
} |
1da177e4c
|
59 60 |
if ((rt->rt_flags & RTCF_BROADCAST) && !sock_flag(sk, SOCK_BROADCAST)) { ip_rt_put(rt); |
3038eeac0
|
61 62 |
err = -EACCES; goto out; |
1da177e4c
|
63 |
} |
c720c7e83
|
64 |
if (!inet->inet_saddr) |
3038eeac0
|
65 |
inet->inet_saddr = fl4->saddr; /* Update source address */ |
719f83585
|
66 |
if (!inet->inet_rcv_saddr) { |
3038eeac0
|
67 |
inet->inet_rcv_saddr = fl4->saddr; |
719f83585
|
68 69 70 |
if (sk->sk_prot->rehash) sk->sk_prot->rehash(sk); } |
3038eeac0
|
71 |
inet->inet_daddr = fl4->daddr; |
c720c7e83
|
72 |
inet->inet_dport = usin->sin_port; |
1da177e4c
|
73 |
sk->sk_state = TCP_ESTABLISHED; |
c720c7e83
|
74 |
inet->inet_id = jiffies; |
1da177e4c
|
75 |
|
d8d1f30b9
|
76 |
sk_dst_set(sk, &rt->dst); |
3038eeac0
|
77 78 79 80 |
err = 0; out: release_sock(sk); return err; |
1da177e4c
|
81 |
} |
1da177e4c
|
82 |
EXPORT_SYMBOL(ip4_datagram_connect); |