1*05b00f60SXin Li /*
2*05b00f60SXin Li * Copyright (c) 1990, 1991, 1993, 1994, 1995, 1996, 1997
3*05b00f60SXin Li * The Regents of the University of California. All rights reserved.
4*05b00f60SXin Li *
5*05b00f60SXin Li * Redistribution and use in source and binary forms, with or without
6*05b00f60SXin Li * modification, are permitted provided that: (1) source code distributions
7*05b00f60SXin Li * retain the above copyright notice and this paragraph in its entirety, (2)
8*05b00f60SXin Li * distributions including binary code include the above copyright notice and
9*05b00f60SXin Li * this paragraph in its entirety in the documentation or other materials
10*05b00f60SXin Li * provided with the distribution, and (3) all advertising materials mentioning
11*05b00f60SXin Li * features or use of this software display the following acknowledgement:
12*05b00f60SXin Li * ``This product includes software developed by the University of California,
13*05b00f60SXin Li * Lawrence Berkeley Laboratory and its contributors.'' Neither the name of
14*05b00f60SXin Li * the University nor the names of its contributors may be used to endorse
15*05b00f60SXin Li * or promote products derived from this software without specific prior
16*05b00f60SXin Li * written permission.
17*05b00f60SXin Li * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR IMPLIED
18*05b00f60SXin Li * WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF
19*05b00f60SXin Li * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
20*05b00f60SXin Li */
21*05b00f60SXin Li
22*05b00f60SXin Li /* \summary: Trivial File Transfer Protocol (TFTP) printer */
23*05b00f60SXin Li
24*05b00f60SXin Li #ifdef HAVE_CONFIG_H
25*05b00f60SXin Li #include <config.h>
26*05b00f60SXin Li #endif
27*05b00f60SXin Li
28*05b00f60SXin Li #include "netdissect-stdinc.h"
29*05b00f60SXin Li
30*05b00f60SXin Li #include "netdissect.h"
31*05b00f60SXin Li #include "extract.h"
32*05b00f60SXin Li
33*05b00f60SXin Li /*
34*05b00f60SXin Li * Trivial File Transfer Protocol (IEN-133)
35*05b00f60SXin Li */
36*05b00f60SXin Li
37*05b00f60SXin Li /*
38*05b00f60SXin Li * Packet types.
39*05b00f60SXin Li */
40*05b00f60SXin Li #define RRQ 01 /* read request */
41*05b00f60SXin Li #define WRQ 02 /* write request */
42*05b00f60SXin Li #define DATA 03 /* data packet */
43*05b00f60SXin Li #define ACK 04 /* acknowledgement */
44*05b00f60SXin Li #define TFTP_ERROR 05 /* error code */
45*05b00f60SXin Li #define OACK 06 /* option acknowledgement */
46*05b00f60SXin Li
47*05b00f60SXin Li /*
48*05b00f60SXin Li * Error codes.
49*05b00f60SXin Li */
50*05b00f60SXin Li #define EUNDEF 0 /* not defined */
51*05b00f60SXin Li #define ENOTFOUND 1 /* file not found */
52*05b00f60SXin Li #define EACCESS 2 /* access violation */
53*05b00f60SXin Li #define ENOSPACE 3 /* disk full or allocation exceeded */
54*05b00f60SXin Li #define EBADOP 4 /* illegal TFTP operation */
55*05b00f60SXin Li #define EBADID 5 /* unknown transfer ID */
56*05b00f60SXin Li #define EEXISTS 6 /* file already exists */
57*05b00f60SXin Li #define ENOUSER 7 /* no such user */
58*05b00f60SXin Li
59*05b00f60SXin Li
60*05b00f60SXin Li /* op code to string mapping */
61*05b00f60SXin Li static const struct tok op2str[] = {
62*05b00f60SXin Li { RRQ, "RRQ" }, /* read request */
63*05b00f60SXin Li { WRQ, "WRQ" }, /* write request */
64*05b00f60SXin Li { DATA, "DATA" }, /* data packet */
65*05b00f60SXin Li { ACK, "ACK" }, /* acknowledgement */
66*05b00f60SXin Li { TFTP_ERROR, "ERROR" }, /* error code */
67*05b00f60SXin Li { OACK, "OACK" }, /* option acknowledgement */
68*05b00f60SXin Li { 0, NULL }
69*05b00f60SXin Li };
70*05b00f60SXin Li
71*05b00f60SXin Li /* error code to string mapping */
72*05b00f60SXin Li static const struct tok err2str[] = {
73*05b00f60SXin Li { EUNDEF, "EUNDEF" }, /* not defined */
74*05b00f60SXin Li { ENOTFOUND, "ENOTFOUND" }, /* file not found */
75*05b00f60SXin Li { EACCESS, "EACCESS" }, /* access violation */
76*05b00f60SXin Li { ENOSPACE, "ENOSPACE" }, /* disk full or allocation exceeded */
77*05b00f60SXin Li { EBADOP, "EBADOP" }, /* illegal TFTP operation */
78*05b00f60SXin Li { EBADID, "EBADID" }, /* unknown transfer ID */
79*05b00f60SXin Li { EEXISTS, "EEXISTS" }, /* file already exists */
80*05b00f60SXin Li { ENOUSER, "ENOUSER" }, /* no such user */
81*05b00f60SXin Li { 0, NULL }
82*05b00f60SXin Li };
83*05b00f60SXin Li
84*05b00f60SXin Li /*
85*05b00f60SXin Li * Print trivial file transfer program requests
86*05b00f60SXin Li */
87*05b00f60SXin Li void
tftp_print(netdissect_options * ndo,const u_char * bp,u_int length)88*05b00f60SXin Li tftp_print(netdissect_options *ndo,
89*05b00f60SXin Li const u_char *bp, u_int length)
90*05b00f60SXin Li {
91*05b00f60SXin Li const char *cp;
92*05b00f60SXin Li u_int opcode;
93*05b00f60SXin Li u_int ui;
94*05b00f60SXin Li
95*05b00f60SXin Li ndo->ndo_protocol = "tftp";
96*05b00f60SXin Li
97*05b00f60SXin Li /* Print protocol */
98*05b00f60SXin Li nd_print_protocol_caps(ndo);
99*05b00f60SXin Li /* Print length */
100*05b00f60SXin Li ND_PRINT(", length %u", length);
101*05b00f60SXin Li
102*05b00f60SXin Li /* Print tftp request type */
103*05b00f60SXin Li if (length < 2)
104*05b00f60SXin Li goto trunc;
105*05b00f60SXin Li opcode = GET_BE_U_2(bp);
106*05b00f60SXin Li cp = tok2str(op2str, "tftp-#%u", opcode);
107*05b00f60SXin Li ND_PRINT(", %s", cp);
108*05b00f60SXin Li /* Bail if bogus opcode */
109*05b00f60SXin Li if (*cp == 't')
110*05b00f60SXin Li return;
111*05b00f60SXin Li bp += 2;
112*05b00f60SXin Li length -= 2;
113*05b00f60SXin Li
114*05b00f60SXin Li switch (opcode) {
115*05b00f60SXin Li
116*05b00f60SXin Li case RRQ:
117*05b00f60SXin Li case WRQ:
118*05b00f60SXin Li if (length == 0)
119*05b00f60SXin Li goto trunc;
120*05b00f60SXin Li ND_PRINT(" ");
121*05b00f60SXin Li /* Print filename */
122*05b00f60SXin Li ND_PRINT("\"");
123*05b00f60SXin Li ui = nd_printztn(ndo, bp, length, ndo->ndo_snapend);
124*05b00f60SXin Li ND_PRINT("\"");
125*05b00f60SXin Li if (ui == 0)
126*05b00f60SXin Li goto trunc;
127*05b00f60SXin Li bp += ui;
128*05b00f60SXin Li length -= ui;
129*05b00f60SXin Li
130*05b00f60SXin Li /* Print the mode - RRQ and WRQ only */
131*05b00f60SXin Li if (length == 0)
132*05b00f60SXin Li goto trunc; /* no mode */
133*05b00f60SXin Li ND_PRINT(" ");
134*05b00f60SXin Li ui = nd_printztn(ndo, bp, length, ndo->ndo_snapend);
135*05b00f60SXin Li if (ui == 0)
136*05b00f60SXin Li goto trunc;
137*05b00f60SXin Li bp += ui;
138*05b00f60SXin Li length -= ui;
139*05b00f60SXin Li
140*05b00f60SXin Li /* Print options, if any */
141*05b00f60SXin Li while (length != 0) {
142*05b00f60SXin Li if (GET_U_1(bp) != '\0')
143*05b00f60SXin Li ND_PRINT(" ");
144*05b00f60SXin Li ui = nd_printztn(ndo, bp, length, ndo->ndo_snapend);
145*05b00f60SXin Li if (ui == 0)
146*05b00f60SXin Li goto trunc;
147*05b00f60SXin Li bp += ui;
148*05b00f60SXin Li length -= ui;
149*05b00f60SXin Li }
150*05b00f60SXin Li break;
151*05b00f60SXin Li
152*05b00f60SXin Li case OACK:
153*05b00f60SXin Li /* Print options */
154*05b00f60SXin Li while (length != 0) {
155*05b00f60SXin Li if (GET_U_1(bp) != '\0')
156*05b00f60SXin Li ND_PRINT(" ");
157*05b00f60SXin Li ui = nd_printztn(ndo, bp, length, ndo->ndo_snapend);
158*05b00f60SXin Li if (ui == 0)
159*05b00f60SXin Li goto trunc;
160*05b00f60SXin Li bp += ui;
161*05b00f60SXin Li length -= ui;
162*05b00f60SXin Li }
163*05b00f60SXin Li break;
164*05b00f60SXin Li
165*05b00f60SXin Li case ACK:
166*05b00f60SXin Li case DATA:
167*05b00f60SXin Li if (length < 2)
168*05b00f60SXin Li goto trunc; /* no block number */
169*05b00f60SXin Li ND_PRINT(" block %u", GET_BE_U_2(bp));
170*05b00f60SXin Li break;
171*05b00f60SXin Li
172*05b00f60SXin Li case TFTP_ERROR:
173*05b00f60SXin Li /* Print error code string */
174*05b00f60SXin Li if (length < 2)
175*05b00f60SXin Li goto trunc; /* no error code */
176*05b00f60SXin Li ND_PRINT(" %s", tok2str(err2str, "tftp-err-#%u \"",
177*05b00f60SXin Li GET_BE_U_2(bp)));
178*05b00f60SXin Li bp += 2;
179*05b00f60SXin Li length -= 2;
180*05b00f60SXin Li /* Print error message string */
181*05b00f60SXin Li if (length == 0)
182*05b00f60SXin Li goto trunc; /* no error message */
183*05b00f60SXin Li ND_PRINT(" \"");
184*05b00f60SXin Li ui = nd_printztn(ndo, bp, length, ndo->ndo_snapend);
185*05b00f60SXin Li ND_PRINT("\"");
186*05b00f60SXin Li if (ui == 0)
187*05b00f60SXin Li goto trunc;
188*05b00f60SXin Li break;
189*05b00f60SXin Li
190*05b00f60SXin Li default:
191*05b00f60SXin Li /* We shouldn't get here */
192*05b00f60SXin Li ND_PRINT("(unknown #%u)", opcode);
193*05b00f60SXin Li break;
194*05b00f60SXin Li }
195*05b00f60SXin Li return;
196*05b00f60SXin Li trunc:
197*05b00f60SXin Li nd_print_trunc(ndo);
198*05b00f60SXin Li }
199