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