1 /*
2 * INET An implementation of the TCP/IP protocol suite for the LINUX
3 * operating system. INET is implemented using the BSD Socket
4 * interface as the means of communication with the user level.
5 *
6 * This file implements the various access functions for the
7 * PROC file system. This is very similar to the IPv4 version,
8 * except it reports the sockets in the INET6 address family.
9 *
10 * Version: $Id: proc.c,v 1.15.2.1 2002/01/24 15:46:07 davem Exp $
11 *
12 * Authors: David S. Miller (davem@caip.rutgers.edu)
13 *
14 * This program is free software; you can redistribute it and/or
15 * modify it under the terms of the GNU General Public License
16 * as published by the Free Software Foundation; either version
17 * 2 of the License, or (at your option) any later version.
18 */
19 #include <linux/sched.h>
20 #include <linux/socket.h>
21 #include <linux/net.h>
22 #include <linux/in6.h>
23 #include <linux/stddef.h>
24 #include <net/sock.h>
25 #include <net/tcp.h>
26 #include <net/transp_v6.h>
27 #include <net/ipv6.h>
28
fold_prot_inuse(struct proto * proto)29 static int fold_prot_inuse(struct proto *proto)
30 {
31 int res = 0;
32 int cpu;
33
34 for (cpu=0; cpu<smp_num_cpus; cpu++)
35 res += proto->stats[cpu_logical_map(cpu)].inuse;
36
37 return res;
38 }
39
afinet6_get_info(char * buffer,char ** start,off_t offset,int length)40 int afinet6_get_info(char *buffer, char **start, off_t offset, int length)
41 {
42 int len = 0;
43 len += sprintf(buffer+len, "TCP6: inuse %d\n",
44 fold_prot_inuse(&tcpv6_prot));
45 len += sprintf(buffer+len, "UDP6: inuse %d\n",
46 fold_prot_inuse(&udpv6_prot));
47 len += sprintf(buffer+len, "RAW6: inuse %d\n",
48 fold_prot_inuse(&rawv6_prot));
49 len += sprintf(buffer+len, "FRAG6: inuse %d memory %d\n",
50 ip6_frag_nqueues, atomic_read(&ip6_frag_mem));
51 *start = buffer + offset;
52 len -= offset;
53 if(len > length)
54 len = length;
55 return len;
56 }
57
58
59 struct snmp6_item
60 {
61 char *name;
62 unsigned long *ptr;
63 int mibsize;
64 } snmp6_list[] = {
65 /* ipv6 mib according to draft-ietf-ipngwg-ipv6-mib-04 */
66 #define SNMP6_GEN(x) { #x , &ipv6_statistics[0].x, sizeof(struct ipv6_mib)/sizeof(unsigned long) }
67 SNMP6_GEN(Ip6InReceives),
68 SNMP6_GEN(Ip6InHdrErrors),
69 SNMP6_GEN(Ip6InTooBigErrors),
70 SNMP6_GEN(Ip6InNoRoutes),
71 SNMP6_GEN(Ip6InAddrErrors),
72 SNMP6_GEN(Ip6InUnknownProtos),
73 SNMP6_GEN(Ip6InTruncatedPkts),
74 SNMP6_GEN(Ip6InDiscards),
75 SNMP6_GEN(Ip6InDelivers),
76 SNMP6_GEN(Ip6OutForwDatagrams),
77 SNMP6_GEN(Ip6OutRequests),
78 SNMP6_GEN(Ip6OutDiscards),
79 SNMP6_GEN(Ip6OutNoRoutes),
80 SNMP6_GEN(Ip6ReasmTimeout),
81 SNMP6_GEN(Ip6ReasmReqds),
82 SNMP6_GEN(Ip6ReasmOKs),
83 SNMP6_GEN(Ip6ReasmFails),
84 SNMP6_GEN(Ip6FragOKs),
85 SNMP6_GEN(Ip6FragFails),
86 SNMP6_GEN(Ip6FragCreates),
87 SNMP6_GEN(Ip6InMcastPkts),
88 SNMP6_GEN(Ip6OutMcastPkts),
89 #undef SNMP6_GEN
90 /* icmpv6 mib according to draft-ietf-ipngwg-ipv6-icmp-mib-02
91
92 Exceptions: {In|Out}AdminProhibs are removed, because I see
93 no good reasons to account them separately
94 of another dest.unreachs.
95 OutErrs is zero identically.
96 OutEchos too.
97 OutRouterAdvertisements too.
98 OutGroupMembQueries too.
99 */
100 #define SNMP6_GEN(x) { #x , &icmpv6_statistics[0].x, sizeof(struct icmpv6_mib)/sizeof(unsigned long) }
101 SNMP6_GEN(Icmp6InMsgs),
102 SNMP6_GEN(Icmp6InErrors),
103 SNMP6_GEN(Icmp6InDestUnreachs),
104 SNMP6_GEN(Icmp6InPktTooBigs),
105 SNMP6_GEN(Icmp6InTimeExcds),
106 SNMP6_GEN(Icmp6InParmProblems),
107 SNMP6_GEN(Icmp6InEchos),
108 SNMP6_GEN(Icmp6InEchoReplies),
109 SNMP6_GEN(Icmp6InGroupMembQueries),
110 SNMP6_GEN(Icmp6InGroupMembResponses),
111 SNMP6_GEN(Icmp6InGroupMembReductions),
112 SNMP6_GEN(Icmp6InRouterSolicits),
113 SNMP6_GEN(Icmp6InRouterAdvertisements),
114 SNMP6_GEN(Icmp6InNeighborSolicits),
115 SNMP6_GEN(Icmp6InNeighborAdvertisements),
116 SNMP6_GEN(Icmp6InRedirects),
117 SNMP6_GEN(Icmp6OutMsgs),
118 SNMP6_GEN(Icmp6OutDestUnreachs),
119 SNMP6_GEN(Icmp6OutPktTooBigs),
120 SNMP6_GEN(Icmp6OutTimeExcds),
121 SNMP6_GEN(Icmp6OutParmProblems),
122 SNMP6_GEN(Icmp6OutEchoReplies),
123 SNMP6_GEN(Icmp6OutRouterSolicits),
124 SNMP6_GEN(Icmp6OutNeighborSolicits),
125 SNMP6_GEN(Icmp6OutNeighborAdvertisements),
126 SNMP6_GEN(Icmp6OutRedirects),
127 SNMP6_GEN(Icmp6OutGroupMembResponses),
128 SNMP6_GEN(Icmp6OutGroupMembReductions),
129 #undef SNMP6_GEN
130 #define SNMP6_GEN(x) { "Udp6" #x , &udp_stats_in6[0].Udp##x, sizeof(struct udp_mib)/sizeof(unsigned long) }
131 SNMP6_GEN(InDatagrams),
132 SNMP6_GEN(NoPorts),
133 SNMP6_GEN(InErrors),
134 SNMP6_GEN(OutDatagrams)
135 #undef SNMP6_GEN
136 };
137
fold_field(unsigned long * ptr,int size)138 static unsigned long fold_field(unsigned long *ptr, int size)
139 {
140 unsigned long res = 0;
141 int i;
142
143 for (i=0; i<smp_num_cpus; i++) {
144 res += ptr[2*cpu_logical_map(i)*size];
145 res += ptr[(2*cpu_logical_map(i)+1)*size];
146 }
147
148 return res;
149 }
150
afinet6_get_snmp(char * buffer,char ** start,off_t offset,int length)151 int afinet6_get_snmp(char *buffer, char **start, off_t offset, int length)
152 {
153 int len = 0;
154 int i;
155
156 for (i=0; i<sizeof(snmp6_list)/sizeof(snmp6_list[0]); i++)
157 len += sprintf(buffer+len, "%-32s\t%ld\n", snmp6_list[i].name,
158 fold_field(snmp6_list[i].ptr, snmp6_list[i].mibsize));
159
160 len -= offset;
161
162 if (len > length)
163 len = length;
164 if(len < 0)
165 len = 0;
166
167 *start = buffer + offset;
168
169 return len;
170 }
171