aboutsummaryrefslogtreecommitdiffstats
path: root/net/ipv4/tcp_diag.c
diff options
context:
space:
mode:
Diffstat (limited to 'net/ipv4/tcp_diag.c')
-rw-r--r--net/ipv4/tcp_diag.c153
1 files changed, 111 insertions, 42 deletions
diff --git a/net/ipv4/tcp_diag.c b/net/ipv4/tcp_diag.c
index b812191b2f5c..b13b71cb9ced 100644
--- a/net/ipv4/tcp_diag.c
+++ b/net/ipv4/tcp_diag.c
@@ -34,6 +34,8 @@
34 34
35#include <linux/tcp_diag.h> 35#include <linux/tcp_diag.h>
36 36
37static const struct inet_diag_handler **inet_diag_table;
38
37struct tcpdiag_entry 39struct tcpdiag_entry
38{ 40{
39 u32 *saddr; 41 u32 *saddr;
@@ -61,18 +63,24 @@ static int tcpdiag_fill(struct sk_buff *skb, struct sock *sk,
61 const struct inet_connection_sock *icsk = inet_csk(sk); 63 const struct inet_connection_sock *icsk = inet_csk(sk);
62 struct tcpdiagmsg *r; 64 struct tcpdiagmsg *r;
63 struct nlmsghdr *nlh; 65 struct nlmsghdr *nlh;
64 struct tcp_info *info = NULL; 66 void *info = NULL;
65 struct tcpdiag_meminfo *minfo = NULL; 67 struct tcpdiag_meminfo *minfo = NULL;
66 unsigned char *b = skb->tail; 68 unsigned char *b = skb->tail;
69 const struct inet_diag_handler *handler;
70
71 handler = inet_diag_table[unlh->nlmsg_type];
72 BUG_ON(handler == NULL);
67 73
68 nlh = NLMSG_PUT(skb, pid, seq, unlh->nlmsg_type, sizeof(*r)); 74 nlh = NLMSG_PUT(skb, pid, seq, unlh->nlmsg_type, sizeof(*r));
69 nlh->nlmsg_flags = nlmsg_flags; 75 nlh->nlmsg_flags = nlmsg_flags;
76
70 r = NLMSG_DATA(nlh); 77 r = NLMSG_DATA(nlh);
71 if (sk->sk_state != TCP_TIME_WAIT) { 78 if (sk->sk_state != TCP_TIME_WAIT) {
72 if (ext & (1<<(TCPDIAG_MEMINFO-1))) 79 if (ext & (1<<(TCPDIAG_MEMINFO-1)))
73 minfo = TCPDIAG_PUT(skb, TCPDIAG_MEMINFO, sizeof(*minfo)); 80 minfo = TCPDIAG_PUT(skb, TCPDIAG_MEMINFO, sizeof(*minfo));
74 if (ext & (1<<(TCPDIAG_INFO-1))) 81 if (ext & (1<<(TCPDIAG_INFO-1)))
75 info = TCPDIAG_PUT(skb, TCPDIAG_INFO, sizeof(*info)); 82 info = TCPDIAG_PUT(skb, TCPDIAG_INFO,
83 handler->idiag_info_size);
76 84
77 if ((ext & (1 << (TCPDIAG_CONG - 1))) && icsk->icsk_ca_ops) { 85 if ((ext & (1 << (TCPDIAG_CONG - 1))) && icsk->icsk_ca_ops) {
78 size_t len = strlen(icsk->icsk_ca_ops->name); 86 size_t len = strlen(icsk->icsk_ca_ops->name);
@@ -155,19 +163,6 @@ static int tcpdiag_fill(struct sk_buff *skb, struct sock *sk,
155 r->tcpdiag_expires = 0; 163 r->tcpdiag_expires = 0;
156 } 164 }
157#undef EXPIRES_IN_MS 165#undef EXPIRES_IN_MS
158 /*
159 * Ahem... for now we'll have some knowledge about TCP -acme
160 * But this is just one of two small exceptions, both in this
161 * function, so lets close our eyes for some 15 lines or so... 8)
162 * -acme
163 */
164 if (sk->sk_protocol == IPPROTO_TCP) {
165 const struct tcp_sock *tp = tcp_sk(sk);
166
167 r->tcpdiag_rqueue = tp->rcv_nxt - tp->copied_seq;
168 r->tcpdiag_wqueue = tp->write_seq - tp->snd_una;
169 } else
170 r->tcpdiag_rqueue = r->tcpdiag_wqueue = 0;
171 166
172 r->tcpdiag_uid = sock_i_uid(sk); 167 r->tcpdiag_uid = sock_i_uid(sk);
173 r->tcpdiag_inode = sock_i_ino(sk); 168 r->tcpdiag_inode = sock_i_ino(sk);
@@ -179,13 +174,7 @@ static int tcpdiag_fill(struct sk_buff *skb, struct sock *sk,
179 minfo->tcpdiag_tmem = atomic_read(&sk->sk_wmem_alloc); 174 minfo->tcpdiag_tmem = atomic_read(&sk->sk_wmem_alloc);
180 } 175 }
181 176
182 /* Ahem... for now we'll have some knowledge about TCP -acme */ 177 handler->idiag_get_info(sk, r, info);
183 if (info) {
184 if (sk->sk_protocol == IPPROTO_TCP)
185 tcp_get_info(sk, info);
186 else
187 memset(info, 0, sizeof(*info));
188 }
189 178
190 if (sk->sk_state < TCP_TIME_WAIT && 179 if (sk->sk_state < TCP_TIME_WAIT &&
191 icsk->icsk_ca_ops && icsk->icsk_ca_ops->get_info) 180 icsk->icsk_ca_ops && icsk->icsk_ca_ops->get_info)
@@ -206,11 +195,13 @@ static int tcpdiag_get_exact(struct sk_buff *in_skb, const struct nlmsghdr *nlh)
206 struct sock *sk; 195 struct sock *sk;
207 struct tcpdiagreq *req = NLMSG_DATA(nlh); 196 struct tcpdiagreq *req = NLMSG_DATA(nlh);
208 struct sk_buff *rep; 197 struct sk_buff *rep;
209 struct inet_hashinfo *hashinfo = &tcp_hashinfo; 198 struct inet_hashinfo *hashinfo;
210#ifdef CONFIG_IP_TCPDIAG_DCCP 199 const struct inet_diag_handler *handler;
211 if (nlh->nlmsg_type == DCCPDIAG_GETSOCK) 200
212 hashinfo = &dccp_hashinfo; 201 handler = inet_diag_table[nlh->nlmsg_type];
213#endif 202 BUG_ON(handler == NULL);
203 hashinfo = handler->idiag_hashinfo;
204
214 if (req->tcpdiag_family == AF_INET) { 205 if (req->tcpdiag_family == AF_INET) {
215 sk = inet_lookup(hashinfo, req->id.tcpdiag_dst[0], 206 sk = inet_lookup(hashinfo, req->id.tcpdiag_dst[0],
216 req->id.tcpdiag_dport, req->id.tcpdiag_src[0], 207 req->id.tcpdiag_dport, req->id.tcpdiag_src[0],
@@ -241,9 +232,10 @@ static int tcpdiag_get_exact(struct sk_buff *in_skb, const struct nlmsghdr *nlh)
241 goto out; 232 goto out;
242 233
243 err = -ENOMEM; 234 err = -ENOMEM;
244 rep = alloc_skb(NLMSG_SPACE(sizeof(struct tcpdiagmsg)+ 235 rep = alloc_skb(NLMSG_SPACE((sizeof(struct tcpdiagmsg) +
245 sizeof(struct tcpdiag_meminfo)+ 236 sizeof(struct tcpdiag_meminfo) +
246 sizeof(struct tcp_info)+64), GFP_KERNEL); 237 handler->idiag_info_size + 64)),
238 GFP_KERNEL);
247 if (!rep) 239 if (!rep)
248 goto out; 240 goto out;
249 241
@@ -603,15 +595,16 @@ static int tcpdiag_dump(struct sk_buff *skb, struct netlink_callback *cb)
603 int i, num; 595 int i, num;
604 int s_i, s_num; 596 int s_i, s_num;
605 struct tcpdiagreq *r = NLMSG_DATA(cb->nlh); 597 struct tcpdiagreq *r = NLMSG_DATA(cb->nlh);
598 const struct inet_diag_handler *handler;
606 struct inet_hashinfo *hashinfo; 599 struct inet_hashinfo *hashinfo;
607 600
601 handler = inet_diag_table[cb->nlh->nlmsg_type];
602 BUG_ON(handler == NULL);
603 hashinfo = handler->idiag_hashinfo;
604
608 s_i = cb->args[1]; 605 s_i = cb->args[1];
609 s_num = num = cb->args[2]; 606 s_num = num = cb->args[2];
610 hashinfo = &tcp_hashinfo; 607
611#ifdef CONFIG_IP_TCPDIAG_DCCP
612 if (cb->nlh->nlmsg_type == DCCPDIAG_GETSOCK)
613 hashinfo = &dccp_hashinfo;
614#endif
615 if (cb->args[0] == 0) { 608 if (cb->args[0] == 0) {
616 if (!(r->tcpdiag_states&(TCPF_LISTEN|TCPF_SYN_RECV))) 609 if (!(r->tcpdiag_states&(TCPF_LISTEN|TCPF_SYN_RECV)))
617 goto skip_listen_ht; 610 goto skip_listen_ht;
@@ -745,13 +738,12 @@ tcpdiag_rcv_msg(struct sk_buff *skb, struct nlmsghdr *nlh)
745 if (!(nlh->nlmsg_flags&NLM_F_REQUEST)) 738 if (!(nlh->nlmsg_flags&NLM_F_REQUEST))
746 return 0; 739 return 0;
747 740
748 if (nlh->nlmsg_type != TCPDIAG_GETSOCK 741 if (nlh->nlmsg_type >= INET_DIAG_GETSOCK_MAX)
749#ifdef CONFIG_IP_TCPDIAG_DCCP
750 && nlh->nlmsg_type != DCCPDIAG_GETSOCK
751#endif
752 )
753 goto err_inval; 742 goto err_inval;
754 743
744 if (inet_diag_table[nlh->nlmsg_type] == NULL)
745 return -ENOENT;
746
755 if (NLMSG_LENGTH(sizeof(struct tcpdiagreq)) > skb->len) 747 if (NLMSG_LENGTH(sizeof(struct tcpdiagreq)) > skb->len)
756 goto err_inval; 748 goto err_inval;
757 749
@@ -803,18 +795,95 @@ static void tcpdiag_rcv(struct sock *sk, int len)
803 } 795 }
804} 796}
805 797
798static void tcp_diag_get_info(struct sock *sk, struct tcpdiagmsg *r,
799 void *_info)
800{
801 const struct tcp_sock *tp = tcp_sk(sk);
802 struct tcp_info *info = _info;
803
804 r->tcpdiag_rqueue = tp->rcv_nxt - tp->copied_seq;
805 r->tcpdiag_wqueue = tp->write_seq - tp->snd_una;
806 if (info != NULL)
807 tcp_get_info(sk, info);
808}
809
810static struct inet_diag_handler tcp_diag_handler = {
811 .idiag_hashinfo = &tcp_hashinfo,
812 .idiag_get_info = tcp_diag_get_info,
813 .idiag_type = TCPDIAG_GETSOCK,
814 .idiag_info_size = sizeof(struct tcp_info),
815};
816
817static DEFINE_SPINLOCK(inet_diag_register_lock);
818
819int inet_diag_register(const struct inet_diag_handler *h)
820{
821 const __u16 type = h->idiag_type;
822 int err = -EINVAL;
823
824 if (type >= INET_DIAG_GETSOCK_MAX)
825 goto out;
826
827 spin_lock(&inet_diag_register_lock);
828 err = -EEXIST;
829 if (inet_diag_table[type] == NULL) {
830 inet_diag_table[type] = h;
831 err = 0;
832 }
833 spin_unlock(&inet_diag_register_lock);
834out:
835 return err;
836}
837EXPORT_SYMBOL_GPL(inet_diag_register);
838
839void inet_diag_unregister(const struct inet_diag_handler *h)
840{
841 const __u16 type = h->idiag_type;
842
843 if (type >= INET_DIAG_GETSOCK_MAX)
844 return;
845
846 spin_lock(&inet_diag_register_lock);
847 inet_diag_table[type] = NULL;
848 spin_unlock(&inet_diag_register_lock);
849
850 synchronize_rcu();
851}
852EXPORT_SYMBOL_GPL(inet_diag_unregister);
853
806static int __init tcpdiag_init(void) 854static int __init tcpdiag_init(void)
807{ 855{
856 const int inet_diag_table_size = (INET_DIAG_GETSOCK_MAX *
857 sizeof(struct inet_diag_handler *));
858 int err = -ENOMEM;
859
860 inet_diag_table = kmalloc(inet_diag_table_size, GFP_KERNEL);
861 if (!inet_diag_table)
862 goto out;
863
864 memset(inet_diag_table, 0, inet_diag_table_size);
865
808 tcpnl = netlink_kernel_create(NETLINK_TCPDIAG, tcpdiag_rcv, 866 tcpnl = netlink_kernel_create(NETLINK_TCPDIAG, tcpdiag_rcv,
809 THIS_MODULE); 867 THIS_MODULE);
810 if (tcpnl == NULL) 868 if (tcpnl == NULL)
811 return -ENOMEM; 869 goto out_free_table;
812 return 0; 870
871 err = inet_diag_register(&tcp_diag_handler);
872 if (err)
873 goto out_sock_release;
874out:
875 return err;
876out_sock_release:
877 sock_release(tcpnl->sk_socket);
878out_free_table:
879 kfree(inet_diag_table);
880 goto out;
813} 881}
814 882
815static void __exit tcpdiag_exit(void) 883static void __exit tcpdiag_exit(void)
816{ 884{
817 sock_release(tcpnl->sk_socket); 885 sock_release(tcpnl->sk_socket);
886 kfree(inet_diag_table);
818} 887}
819 888
820module_init(tcpdiag_init); 889module_init(tcpdiag_init);