xref: /freebsd/contrib/bsnmp/snmp_mibII/mibII_udp.c (revision 6b3455a7665208c366849f0b2b3bc916fb97516e)
1 /*
2  * Copyright (c) 2001-2003
3  *	Fraunhofer Institute for Open Communication Systems (FhG Fokus).
4  *	All rights reserved.
5  *
6  * Author: Harti Brandt <harti@freebsd.org>
7  *
8  * Redistribution of this software and documentation and use in source and
9  * binary forms, with or without modification, are permitted provided that
10  * the following conditions are met:
11  *
12  * 1. Redistributions of source code or documentation must retain the above
13  *    copyright notice, this list of conditions and the following disclaimer.
14  * 2. Redistributions in binary form must reproduce the above copyright
15  *    notice, this list of conditions and the following disclaimer in the
16  *    documentation and/or other materials provided with the distribution.
17  * 3. Neither the name of the Institute nor the names of its contributors
18  *    may be used to endorse or promote products derived from this software
19  *    without specific prior written permission.
20  *
21  * THIS SOFTWARE AND DOCUMENTATION IS PROVIDED BY FRAUNHOFER FOKUS
22  * AND ITS CONTRIBUTORS ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES,
23  * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
24  * FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL
25  * FRAUNHOFER FOKUS OR ITS CONTRIBUTORS  BE LIABLE FOR ANY DIRECT, INDIRECT,
26  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
27  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
28  * OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
29  * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
30  * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
31  * EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
32  *
33  * $Begemot: bsnmp/snmp_mibII/mibII_udp.c,v 1.5 2003/12/03 10:01:19 hbb Exp $
34  *
35  * udp
36  */
37 #include "mibII.h"
38 #include "mibII_oid.h"
39 #include <sys/socketvar.h>
40 #include <netinet/in_pcb.h>
41 #include <netinet/udp.h>
42 #include <netinet/ip_var.h>
43 #include <netinet/udp_var.h>
44 
45 struct udp_index {
46 	struct asn_oid	index;
47 	struct xinpcb	*inp;
48 };
49 
50 static u_int32_t udp_tick;
51 static struct udpstat udpstat;
52 static struct xinpgen *xinpgen;
53 static size_t xinpgen_len;
54 static u_int udp_total;
55 
56 static u_int oidnum;
57 static struct udp_index *udpoids;
58 
59 static int
60 udp_compare(const void *p1, const void *p2)
61 {
62 	const struct udp_index *t1 = p1;
63 	const struct udp_index *t2 = p2;
64 
65 	return (asn_compare_oid(&t1->index, &t2->index));
66 }
67 
68 static int
69 fetch_udp(void)
70 {
71 	size_t len;
72 	struct xinpgen *ptr;
73 	struct xinpcb *inp;
74 	struct udp_index *oid;
75 	in_addr_t inaddr;
76 
77 	len = sizeof(udpstat);
78 	if (sysctlbyname("net.inet.udp.stats", &udpstat, &len, NULL, 0) == -1) {
79 		syslog(LOG_ERR, "net.inet.udp.stats: %m");
80 		return (-1);
81 	}
82 	if (len != sizeof(udpstat)) {
83 		syslog(LOG_ERR, "net.inet.udp.stats: wrong size");
84 		return (-1);
85 	}
86 
87 	udp_tick = get_ticks();
88 
89 	len = 0;
90 	if (sysctlbyname("net.inet.udp.pcblist", NULL, &len, NULL, 0) == -1) {
91 		syslog(LOG_ERR, "net.inet.udp.pcblist: %m");
92 		return (-1);
93 	}
94 	if (len > xinpgen_len) {
95 		if ((ptr = realloc(xinpgen, len)) == NULL) {
96 			syslog(LOG_ERR, "%zu: %m", len);
97 			return (-1);
98 		}
99 		xinpgen = ptr;
100 		xinpgen_len = len;
101 	}
102 	if (sysctlbyname("net.inet.udp.pcblist", xinpgen, &len, NULL, 0) == -1) {
103 		syslog(LOG_ERR, "net.inet.udp.pcblist: %m");
104 		return (-1);
105 	}
106 
107 	udp_total = 0;
108 	for (ptr = (struct xinpgen *)(void *)((char *)xinpgen + xinpgen->xig_len);
109 	     ptr->xig_len > sizeof(struct xinpgen);
110              ptr = (struct xinpgen *)(void *)((char *)ptr + ptr->xig_len)) {
111 		inp = (struct xinpcb *)ptr;
112 		if (inp->xi_inp.inp_gencnt > xinpgen->xig_gen ||
113 		    (inp->xi_inp.inp_vflag & INP_IPV4) == 0)
114 			continue;
115 
116 		udp_total++;
117 	}
118 
119 	if (oidnum < udp_total) {
120 		oid = realloc(udpoids, udp_total * sizeof(udpoids[0]));
121 		if (oid == NULL) {
122 			free(udpoids);
123 			oidnum = 0;
124 			return (0);
125 		}
126 		udpoids = oid;
127 		oidnum = udp_total;
128 	}
129 
130 	oid = udpoids;
131 	for (ptr = (struct xinpgen *)(void *)((char *)xinpgen + xinpgen->xig_len);
132 	     ptr->xig_len > sizeof(struct xinpgen);
133              ptr = (struct xinpgen *)(void *)((char *)ptr + ptr->xig_len)) {
134 		inp = (struct xinpcb *)ptr;
135 		if (inp->xi_inp.inp_gencnt > xinpgen->xig_gen ||
136 		    (inp->xi_inp.inp_vflag & INP_IPV4) == 0)
137 			continue;
138 		oid->inp = inp;
139 		oid->index.len = 5;
140 		inaddr = ntohl(inp->xi_inp.inp_laddr.s_addr);
141 		oid->index.subs[0] = (inaddr >> 24) & 0xff;
142 		oid->index.subs[1] = (inaddr >> 16) & 0xff;
143 		oid->index.subs[2] = (inaddr >>  8) & 0xff;
144 		oid->index.subs[3] = (inaddr >>  0) & 0xff;
145 		oid->index.subs[4] = ntohs(inp->xi_inp.inp_lport);
146 		oid++;
147 	}
148 
149 	qsort(udpoids, udp_total, sizeof(udpoids[0]), udp_compare);
150 
151 	return (0);
152 }
153 
154 int
155 op_udp(struct snmp_context *ctx __unused, struct snmp_value *value,
156     u_int sub, u_int iidx __unused, enum snmp_op op)
157 {
158 	switch (op) {
159 
160 	  case SNMP_OP_GETNEXT:
161 		abort();
162 
163 	  case SNMP_OP_GET:
164 		break;
165 
166 	  case SNMP_OP_SET:
167 		return (SNMP_ERR_NOT_WRITEABLE);
168 
169 	  case SNMP_OP_ROLLBACK:
170 	  case SNMP_OP_COMMIT:
171 		abort();
172 	}
173 
174 	if (udp_tick < this_tick)
175 		if (fetch_udp() == -1)
176 			return (SNMP_ERR_GENERR);
177 
178 	switch (value->var.subs[sub - 1]) {
179 
180 	  case LEAF_udpInDatagrams:
181 		value->v.uint32 = udpstat.udps_ipackets;
182 		break;
183 
184 	  case LEAF_udpNoPorts:
185 		value->v.uint32 = udpstat.udps_noport +
186 		    udpstat.udps_noportbcast +
187 		    udpstat.udps_noportmcast;
188 		break;
189 
190 	  case LEAF_udpInErrors:
191 		value->v.uint32 = udpstat.udps_hdrops +
192 		    udpstat.udps_badsum +
193 		    udpstat.udps_badlen +
194 		    udpstat.udps_fullsock;
195 		break;
196 
197 	  case LEAF_udpOutDatagrams:
198 		value->v.uint32 = udpstat.udps_opackets;
199 		break;
200 	}
201 	return (SNMP_ERR_NOERROR);
202 }
203 
204 int
205 op_udptable(struct snmp_context *ctx __unused, struct snmp_value *value,
206     u_int sub, u_int iidx __unused, enum snmp_op op)
207 {
208 	u_int i;
209 
210 	if (udp_tick < this_tick)
211 		if (fetch_udp() == -1)
212 			return (SNMP_ERR_GENERR);
213 
214 	switch (op) {
215 
216 	  case SNMP_OP_GETNEXT:
217 		for (i = 0; i < udp_total; i++)
218 			if (index_compare(&value->var, sub, &udpoids[i].index) < 0)
219 				break;
220 		if (i == udp_total)
221 			return (SNMP_ERR_NOSUCHNAME);
222 		index_append(&value->var, sub, &udpoids[i].index);
223 		break;
224 
225 	  case SNMP_OP_GET:
226 		for (i = 0; i < udp_total; i++)
227 			if (index_compare(&value->var, sub, &udpoids[i].index) == 0)
228 				break;
229 		if (i == udp_total)
230 			return (SNMP_ERR_NOSUCHNAME);
231 		break;
232 
233 	  case SNMP_OP_SET:
234 		return (SNMP_ERR_NOT_WRITEABLE);
235 
236 	  case SNMP_OP_ROLLBACK:
237 	  case SNMP_OP_COMMIT:
238 	  default:
239 		abort();
240 	}
241 
242 	switch (value->var.subs[sub - 1]) {
243 
244 	  case LEAF_udpLocalAddress:
245 		value->v.ipaddress[0] = udpoids[i].index.subs[0];
246 		value->v.ipaddress[1] = udpoids[i].index.subs[1];
247 		value->v.ipaddress[2] = udpoids[i].index.subs[2];
248 		value->v.ipaddress[3] = udpoids[i].index.subs[3];
249 		break;
250 
251 	  case LEAF_udpLocalPort:
252 		value->v.integer = udpoids[i].index.subs[4];
253 		break;
254 
255 	}
256 	return (SNMP_ERR_NOERROR);
257 }
258