xref: /freebsd/contrib/tcpdump/print-telnet.c (revision 282a3889ebf826db9839be296ff1dd903f6d6d6e)
1 /*	$NetBSD: print-telnet.c,v 1.2 1999/10/11 12:40:12 sjg Exp $ 	*/
2 
3 /*-
4  * Copyright (c) 1997, 1998 The NetBSD Foundation, Inc.
5  * All rights reserved.
6  *
7  * This code is derived from software contributed to The NetBSD Foundation
8  * by Simon J. Gerraty.
9  *
10  * Redistribution and use in source and binary forms, with or without
11  * modification, are permitted provided that the following conditions
12  * are met:
13  * 1. Redistributions of source code must retain the above copyright
14  *    notice, this list of conditions and the following disclaimer.
15  * 2. Redistributions in binary form must reproduce the above copyright
16  *    notice, this list of conditions and the following disclaimer in the
17  *    documentation and/or other materials provided with the distribution.
18  * 3. All advertising materials mentioning features or use of this software
19  *    must display the following acknowledgement:
20  *        This product includes software developed by the NetBSD
21  *        Foundation, Inc. and its contributors.
22  * 4. Neither the name of The NetBSD Foundation nor the names of its
23  *    contributors may be used to endorse or promote products derived
24  *    from this software without specific prior written permission.
25  *
26  * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
27  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
28  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
29  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
30  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
31  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
32  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
33  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
34  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
35  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
36  * POSSIBILITY OF SUCH DAMAGE.
37  */
38 /*
39  *      @(#)Copyright (c) 1994, Simon J. Gerraty.
40  *
41  *      This is free software.  It comes with NO WARRANTY.
42  *      Permission to use, modify and distribute this source code
43  *      is granted subject to the following conditions.
44  *      1/ that the above copyright notice and this notice
45  *      are preserved in all copies.
46  */
47 
48 #ifdef HAVE_CONFIG_H
49 #include "config.h"
50 #endif
51 
52 #ifndef lint
53 static const char rcsid[] _U_ =
54      "@(#) $Header: /tcpdump/master/tcpdump/print-telnet.c,v 1.24 2003/12/29 11:05:10 hannes Exp $";
55 #endif
56 
57 #include <tcpdump-stdinc.h>
58 
59 #include <stdio.h>
60 #include <stdlib.h>
61 #include <string.h>
62 
63 #include "interface.h"
64 #include "addrtoname.h"
65 
66 #define TELCMDS
67 #define TELOPTS
68 #include "telnet.h"
69 
70 /* normal */
71 static const char *cmds[] = {
72 	"IS", "SEND", "INFO",
73 };
74 
75 /* 37: Authentication */
76 static const char *authcmd[] = {
77 	"IS", "SEND", "REPLY", "NAME",
78 };
79 static const char *authtype[] = {
80 	"NULL", "KERBEROS_V4", "KERBEROS_V5", "SPX", "MINK",
81 	"SRP", "RSA", "SSL", NULL, NULL,
82 	"LOKI", "SSA", "KEA_SJ", "KEA_SJ_INTEG", "DSS",
83 	"NTLM",
84 };
85 
86 /* 38: Encryption */
87 static const char *enccmd[] = {
88 	"IS", "SUPPORT", "REPLY", "START", "END",
89 	"REQUEST-START", "REQUEST-END", "END_KEYID", "DEC_KEYID",
90 };
91 static const char *enctype[] = {
92 	"NULL", "DES_CFB64", "DES_OFB64", "DES3_CFB64", "DES3_OFB64",
93 	NULL, "CAST5_40_CFB64", "CAST5_40_OFB64", "CAST128_CFB64", "CAST128_OFB64",
94 };
95 
96 #define STR_OR_ID(x, tab) \
97 	(((x) < sizeof(tab)/sizeof(tab[0]) && tab[(x)]) ? tab[(x)] : numstr(x))
98 
99 static char *
100 numstr(int x)
101 {
102 	static char buf[20];
103 
104 	snprintf(buf, sizeof(buf), "%#x", x);
105 	return buf;
106 }
107 
108 /* sp points to IAC byte */
109 static int
110 telnet_parse(const u_char *sp, u_int length, int print)
111 {
112 	int i, x;
113 	u_int c;
114 	const u_char *osp, *p;
115 #define FETCH(c, sp, length) \
116 	do { \
117 		if (length < 1) \
118 			goto pktend; \
119 		TCHECK(*sp); \
120 		c = *sp++; \
121 		length--; \
122 	} while (0)
123 
124 	osp = sp;
125 
126 	FETCH(c, sp, length);
127 	if (c != IAC)
128 		goto pktend;
129 	FETCH(c, sp, length);
130 	if (c == IAC) {		/* <IAC><IAC>! */
131 		if (print)
132 			printf("IAC IAC");
133 		goto done;
134 	}
135 
136 	i = c - TELCMD_FIRST;
137 	if (i < 0 || i > IAC - TELCMD_FIRST)
138 		goto pktend;
139 
140 	switch (c) {
141 	case DONT:
142 	case DO:
143 	case WONT:
144 	case WILL:
145 	case SB:
146 		/* DONT/DO/WONT/WILL x */
147 		FETCH(x, sp, length);
148 		if (x >= 0 && x < NTELOPTS) {
149 			if (print)
150 				(void)printf("%s %s", telcmds[i], telopts[x]);
151 		} else {
152 			if (print)
153 				(void)printf("%s %#x", telcmds[i], x);
154 		}
155 		if (c != SB)
156 			break;
157 		/* IAC SB .... IAC SE */
158 		p = sp;
159 		while (length > (u_int)(p + 1 - sp)) {
160 			if (p[0] == IAC && p[1] == SE)
161 				break;
162 			p++;
163 		}
164 		if (*p != IAC)
165 			goto pktend;
166 
167 		switch (x) {
168 		case TELOPT_AUTHENTICATION:
169 			if (p <= sp)
170 				break;
171 			FETCH(c, sp, length);
172 			if (print)
173 				(void)printf(" %s", STR_OR_ID(c, authcmd));
174 			if (p <= sp)
175 				break;
176 			FETCH(c, sp, length);
177 			if (print)
178 				(void)printf(" %s", STR_OR_ID(c, authtype));
179 			break;
180 		case TELOPT_ENCRYPT:
181 			if (p <= sp)
182 				break;
183 			FETCH(c, sp, length);
184 			if (print)
185 				(void)printf(" %s", STR_OR_ID(c, enccmd));
186 			if (p <= sp)
187 				break;
188 			FETCH(c, sp, length);
189 			if (print)
190 				(void)printf(" %s", STR_OR_ID(c, enctype));
191 			break;
192 		default:
193 			if (p <= sp)
194 				break;
195 			FETCH(c, sp, length);
196 			if (print)
197 				(void)printf(" %s", STR_OR_ID(c, cmds));
198 			break;
199 		}
200 		while (p > sp) {
201 			FETCH(x, sp, length);
202 			if (print)
203 				(void)printf(" %#x", x);
204 		}
205 		/* terminating IAC SE */
206 		if (print)
207 			(void)printf(" SE");
208 		sp += 2;
209 		length -= 2;
210 		break;
211 	default:
212 		if (print)
213 			(void)printf("%s", telcmds[i]);
214 		goto done;
215 	}
216 
217 done:
218 	return sp - osp;
219 
220 trunc:
221 	(void)printf("[|telnet]");
222 pktend:
223 	return -1;
224 #undef FETCH
225 }
226 
227 void
228 telnet_print(const u_char *sp, u_int length)
229 {
230 	int first = 1;
231 	const u_char *osp;
232 	int l;
233 
234 	osp = sp;
235 
236 	while (length > 0 && *sp == IAC) {
237 		l = telnet_parse(sp, length, 0);
238 		if (l < 0)
239 			break;
240 
241 		/*
242 		 * now print it
243 		 */
244 		if (Xflag && 2 < vflag) {
245 			if (first)
246 				printf("\nTelnet:");
247 			hex_print_with_offset("\n", sp, l, sp - osp);
248 			if (l > 8)
249 				printf("\n\t\t\t\t");
250 			else
251 				printf("%*s\t", (8 - l) * 3, "");
252 		} else
253 			printf("%s", (first) ? " [telnet " : ", ");
254 
255 		(void)telnet_parse(sp, length, 1);
256 		first = 0;
257 
258 		sp += l;
259 		length -= l;
260 	}
261 	if (!first) {
262 		if (Xflag && 2 < vflag)
263 			printf("\n");
264 		else
265 			printf("]");
266 	}
267 }
268