xref: /freebsd/usr.bin/systat/sctp.c (revision 2e3507c25e42292b45a5482e116d278f5515d04d)
1 /*-
2  * Copyright (c) 2015
3  * The Regents of the University of California.  All rights reserved.
4  * Michael Tuexen.  All rights reserved.
5  *
6  * Redistribution and use in source and binary forms, with or without
7  * modification, are permitted provided that the following conditions
8  * are met:
9  * 1. Redistributions of source code must retain the above copyright
10  *    notice, this list of conditions and the following disclaimer.
11  * 2. Redistributions in binary form must reproduce the above copyright
12  *    notice, this list of conditions and the following disclaimer in the
13  *    documentation and/or other materials provided with the distribution.
14  * 3. Neither the name of the University nor the names of its contributors
15  *    may be used to endorse or promote products derived from this software
16  *    without specific prior written permission.
17  *
18  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
19  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
20  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
21  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
22  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
23  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
24  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
25  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
26  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
27  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
28  * SUCH DAMAGE.
29  */
30 
31 #include <sys/param.h>
32 #include <sys/types.h>
33 #include <sys/socket.h>
34 #include <sys/sysctl.h>
35 
36 #include <netinet/sctp.h>
37 
38 #include <stdlib.h>
39 #include <string.h>
40 
41 #include "systat.h"
42 #include "extern.h"
43 #include "mode.h"
44 
45 static struct sctpstat curstat, initstat, oldstat;
46 
47 /*-
48 --0         1         2         3         4         5         6         7
49 --0123456789012345678901234567890123456789012345678901234567890123456789012345
50 00             SCTP Associations                     SCTP Packets
51 01999999999999 associations initiated   999999999999 packets sent
52 02999999999999 associations accepted    999999999999 packets received
53 03999999999999 associations restarted   999999999999 - out of the blue
54 04999999999999 associations terminated  999999999999 - bad vtag
55 05999999999999 associations aborted     999999999999 - bad crc32c
56 06
57 07             SCTP Timers                           SCTP Chunks
58 08999999999999 init timeouts            999999999999 control chunks sent
59 09999999999999 cookie timeouts          999999999999 data chunks sent
60 10999999999999 data timeouts            999999999999 - ordered
61 11999999999999 delayed sack timeouts    999999999999 - unordered
62 12999999999999 shutdown timeouts        999999999999 control chunks received
63 13999999999999 shutdown-ack timeouts    999999999999 data chunks received
64 14999999999999 shutdown guard timeouts  999999999999 - ordered
65 15999999999999 heartbeat timeouts       999999999999 - unordered
66 16999999999999 path MTU timeouts
67 17999999999999 autoclose timeouts                    SCTP user messages
68 18999999999999 asconf timeouts          999999999999 fragmented
69 19999999999999 stream reset timeouts    999999999999 reassembled
70 --0123456789012345678901234567890123456789012345678901234567890123456789012345
71 --0         1         2         3         4         5         6         7
72 */
73 
74 WINDOW *
75 opensctp(void)
76 {
77 	return (subwin(stdscr, LINES-3-1, 0, MAINWIN_ROW, 0));
78 }
79 
80 void
81 closesctp(WINDOW *w)
82 {
83 	if (w != NULL) {
84 		wclear(w);
85 		wrefresh(w);
86 		delwin(w);
87 	}
88 }
89 
90 void
91 labelsctp(void)
92 {
93 	wmove(wnd, 0, 0); wclrtoeol(wnd);
94 #define L(row, str) mvwprintw(wnd, row, 13, str)
95 #define R(row, str) mvwprintw(wnd, row, 51, str);
96 	L(0, "SCTP Associations");		R(0, "SCTP Packets");
97 	L(1, "associations initiated");		R(1, "packets sent");
98 	L(2, "associations accepted");		R(2, "packets received");
99 	L(3, "associations restarted");		R(3, "- out of the blue");
100 	L(4, "associations terminated");	R(4, "- bad vtag");
101 	L(5, "associations aborted");		R(5, "- bad crc32c");
102 
103 	L(7, "SCTP Timers");			R(7, "SCTP Chunks");
104 	L(8, "init timeouts");			R(8, "control chunks sent");
105 	L(9, "cookie timeouts");		R(9, "data chunks sent");
106 	L(10, "data timeouts");			R(10, "- ordered");
107 	L(11, "delayed sack timeouts");		R(11, "- unordered");
108 	L(12, "shutdown timeouts");		R(12, "control chunks received");
109 	L(13, "shutdown-ack timeouts");		R(13, "data chunks received");
110 	L(14, "shutdown guard timeouts");	R(14, "- ordered");
111 	L(15, "heartbeat timeouts");		R(15, "- unordered");
112 	L(16, "path MTU timeouts");
113 	L(17, "autoclose timeouts");		R(17, "SCTP User Messages");
114 	L(18, "asconf timeouts");		R(18, "fragmented");
115 	L(19, "stream reset timeouts");		R(19, "reassembled");
116 #undef L
117 #undef R
118 }
119 
120 static void
121 domode(struct sctpstat *ret)
122 {
123 	const struct sctpstat *sub;
124 	int divisor = 1;
125 
126 	switch(currentmode) {
127 	case display_RATE:
128 		sub = &oldstat;
129 		divisor = (delay > 1000000) ? delay / 1000000 : 1;
130 		break;
131 	case display_DELTA:
132 		sub = &oldstat;
133 		break;
134 	case display_SINCE:
135 		sub = &initstat;
136 		break;
137 	default:
138 		*ret = curstat;
139 		return;
140 	}
141 #define DO(stat) ret->stat = (curstat.stat - sub->stat) / divisor
142 	DO(sctps_currestab);
143 	DO(sctps_activeestab);
144 	DO(sctps_restartestab);
145 	DO(sctps_collisionestab);
146 	DO(sctps_passiveestab);
147 	DO(sctps_aborted);
148 	DO(sctps_shutdown);
149 	DO(sctps_outoftheblue);
150 	DO(sctps_checksumerrors);
151 	DO(sctps_outcontrolchunks);
152 	DO(sctps_outorderchunks);
153 	DO(sctps_outunorderchunks);
154 	DO(sctps_incontrolchunks);
155 	DO(sctps_inorderchunks);
156 	DO(sctps_inunorderchunks);
157 	DO(sctps_fragusrmsgs);
158 	DO(sctps_reasmusrmsgs);
159 	DO(sctps_outpackets);
160 	DO(sctps_inpackets);
161 
162 	DO(sctps_recvpackets);
163 	DO(sctps_recvdatagrams);
164 	DO(sctps_recvpktwithdata);
165 	DO(sctps_recvsacks);
166 	DO(sctps_recvdata);
167 	DO(sctps_recvdupdata);
168 	DO(sctps_recvheartbeat);
169 	DO(sctps_recvheartbeatack);
170 	DO(sctps_recvecne);
171 	DO(sctps_recvauth);
172 	DO(sctps_recvauthmissing);
173 	DO(sctps_recvivalhmacid);
174 	DO(sctps_recvivalkeyid);
175 	DO(sctps_recvauthfailed);
176 	DO(sctps_recvexpress);
177 	DO(sctps_recvexpressm);
178 	DO(sctps_recvswcrc);
179 	DO(sctps_recvhwcrc);
180 
181 	DO(sctps_sendpackets);
182 	DO(sctps_sendsacks);
183 	DO(sctps_senddata);
184 	DO(sctps_sendretransdata);
185 	DO(sctps_sendfastretrans);
186 	DO(sctps_sendmultfastretrans);
187 	DO(sctps_sendheartbeat);
188 	DO(sctps_sendecne);
189 	DO(sctps_sendauth);
190 	DO(sctps_senderrors);
191 	DO(sctps_sendswcrc);
192 	DO(sctps_sendhwcrc);
193 
194 	DO(sctps_pdrpfmbox);
195 	DO(sctps_pdrpfehos);
196 	DO(sctps_pdrpmbda);
197 	DO(sctps_pdrpmbct);
198 	DO(sctps_pdrpbwrpt);
199 	DO(sctps_pdrpcrupt);
200 	DO(sctps_pdrpnedat);
201 	DO(sctps_pdrppdbrk);
202 	DO(sctps_pdrptsnnf);
203 	DO(sctps_pdrpdnfnd);
204 	DO(sctps_pdrpdiwnp);
205 	DO(sctps_pdrpdizrw);
206 	DO(sctps_pdrpbadd);
207 	DO(sctps_pdrpmark);
208 
209 	DO(sctps_timoiterator);
210 	DO(sctps_timodata);
211 	DO(sctps_timowindowprobe);
212 	DO(sctps_timoinit);
213 	DO(sctps_timosack);
214 	DO(sctps_timoshutdown);
215 	DO(sctps_timoheartbeat);
216 	DO(sctps_timocookie);
217 	DO(sctps_timosecret);
218 	DO(sctps_timopathmtu);
219 	DO(sctps_timoshutdownack);
220 	DO(sctps_timoshutdownguard);
221 	DO(sctps_timostrmrst);
222 	DO(sctps_timoearlyfr);
223 	DO(sctps_timoasconf);
224 	DO(sctps_timodelprim);
225 	DO(sctps_timoautoclose);
226 	DO(sctps_timoassockill);
227 	DO(sctps_timoinpkill);
228 
229 	DO(sctps_hdrops);
230 	DO(sctps_badsum);
231 	DO(sctps_noport);
232 	DO(sctps_badvtag);
233 	DO(sctps_badsid);
234 	DO(sctps_nomem);
235 	DO(sctps_fastretransinrtt);
236 	DO(sctps_markedretrans);
237 	DO(sctps_naglesent);
238 	DO(sctps_naglequeued);
239 	DO(sctps_maxburstqueued);
240 	DO(sctps_ifnomemqueued);
241 	DO(sctps_windowprobed);
242 	DO(sctps_lowlevelerr);
243 	DO(sctps_lowlevelerrusr);
244 	DO(sctps_datadropchklmt);
245 	DO(sctps_datadroprwnd);
246 	DO(sctps_ecnereducedcwnd);
247 	DO(sctps_vtagexpress);
248 	DO(sctps_vtagbogus);
249 	DO(sctps_primary_randry);
250 	DO(sctps_cmt_randry);
251 	DO(sctps_slowpath_sack);
252 	DO(sctps_wu_sacks_sent);
253 	DO(sctps_sends_with_flags);
254 	DO(sctps_sends_with_unord);
255 	DO(sctps_sends_with_eof);
256 	DO(sctps_sends_with_abort);
257 	DO(sctps_protocol_drain_calls);
258 	DO(sctps_protocol_drains_done);
259 	DO(sctps_read_peeks);
260 	DO(sctps_cached_chk);
261 	DO(sctps_cached_strmoq);
262 	DO(sctps_left_abandon);
263 	DO(sctps_send_burst_avoid);
264 	DO(sctps_send_cwnd_avoid);
265 	DO(sctps_fwdtsn_map_over);
266 	DO(sctps_queue_upd_ecne);
267 #undef DO
268 }
269 
270 void
271 showsctp(void)
272 {
273 	struct sctpstat stats;
274 
275 	memset(&stats, 0, sizeof stats);
276 	domode(&stats);
277 
278 #define DO(stat, row, col) \
279 	mvwprintw(wnd, row, col, "%12lu", stats.stat)
280 #define	L(row, stat) DO(stat, row, 0)
281 #define	R(row, stat) DO(stat, row, 38)
282 	L(1, sctps_activeestab);	R(1, sctps_outpackets);
283 	L(2, sctps_passiveestab);	R(2, sctps_inpackets);
284 	L(3, sctps_restartestab);	R(3, sctps_outoftheblue);
285 	L(4, sctps_shutdown);		R(4, sctps_badvtag);
286 	L(5, sctps_aborted);		R(5, sctps_checksumerrors);
287 
288 
289 	L(8, sctps_timoinit);		R(8, sctps_outcontrolchunks);
290 	L(9, sctps_timocookie);		R(9, sctps_senddata);
291 	L(10, sctps_timodata);		R(10, sctps_outorderchunks);
292 	L(11, sctps_timosack);		R(11, sctps_outunorderchunks);
293 	L(12, sctps_timoshutdown);	R(12, sctps_incontrolchunks);
294 	L(13, sctps_timoshutdownack);	R(13, sctps_recvdata);
295 	L(14, sctps_timoshutdownguard);	R(14, sctps_inorderchunks);
296 	L(15, sctps_timoheartbeat);	R(15, sctps_inunorderchunks);
297 	L(16, sctps_timopathmtu);
298 	L(17, sctps_timoautoclose);
299 	L(18, sctps_timoasconf);	R(18, sctps_fragusrmsgs);
300 	L(19, sctps_timostrmrst);	R(19, sctps_reasmusrmsgs);
301 #undef DO
302 #undef L
303 #undef R
304 }
305 
306 int
307 initsctp(void)
308 {
309 	size_t len;
310 	const char *name = "net.inet.sctp.stats";
311 
312 	len = 0;
313 	if (sysctlbyname(name, NULL, &len, NULL, 0) < 0) {
314 		error("sysctl getting sctpstat size failed");
315 		return 0;
316 	}
317 	if (len > sizeof curstat) {
318 		error("sctpstat structure has grown--recompile systat!");
319 		return 0;
320 	}
321 	if (sysctlbyname(name, &initstat, &len, NULL, 0) < 0) {
322 		error("sysctl getting sctpstat failed");
323 		return 0;
324 	}
325 	oldstat = initstat;
326 	return 1;
327 }
328 
329 void
330 resetsctp(void)
331 {
332 	size_t len;
333 	const char *name = "net.inet.sctp.stats";
334 
335 	len = sizeof initstat;
336 	if (sysctlbyname(name, &initstat, &len, NULL, 0) < 0) {
337 		error("sysctl getting sctpstat failed");
338 	}
339 	oldstat = initstat;
340 }
341 
342 void
343 fetchsctp(void)
344 {
345 	size_t len;
346 	const char *name = "net.inet.sctp.stats";
347 
348 	oldstat = curstat;
349 	len = sizeof curstat;
350 	if (sysctlbyname(name, &curstat, &len, NULL, 0) < 0) {
351 		error("sysctl getting sctpstat failed");
352 	}
353 	return;
354 }
355