xref: /dflybsd-src/contrib/tcpdump/print-resp.c (revision 59c07fbdf8168fa08c76c515186d561b5a92690c)
1411677aeSAaron LI /*
2411677aeSAaron LI  * Copyright (c) 2015 The TCPDUMP project
3411677aeSAaron LI  * All rights reserved.
4411677aeSAaron LI  *
5411677aeSAaron LI  * Redistribution and use in source and binary forms, with or without
6411677aeSAaron LI  * modification, are permitted provided that the following conditions
7411677aeSAaron LI  * are met:
8411677aeSAaron LI  * 1. Redistributions of source code must retain the above copyright
9411677aeSAaron LI  *    notice, this list of conditions and the following disclaimer.
10411677aeSAaron LI  * 2. Redistributions in binary form must reproduce the above copyright
11411677aeSAaron LI  *    notice, this list of conditions and the following disclaimer in the
12411677aeSAaron LI  *    documentation and/or other materials provided with the distribution.
13411677aeSAaron LI  *
14411677aeSAaron LI  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
15411677aeSAaron LI  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
16411677aeSAaron LI  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
17411677aeSAaron LI  * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
18411677aeSAaron LI  * COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
19411677aeSAaron LI  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
20411677aeSAaron LI  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
21411677aeSAaron LI  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
22411677aeSAaron LI  * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
23411677aeSAaron LI  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
24411677aeSAaron LI  * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
25411677aeSAaron LI  * POSSIBILITY OF SUCH DAMAGE.
26411677aeSAaron LI  *
27411677aeSAaron LI  * Initial contribution by Andrew Darqui (andrew.darqui@gmail.com).
28411677aeSAaron LI  */
29411677aeSAaron LI 
30411677aeSAaron LI /* \summary: REdis Serialization Protocol (RESP) printer */
31411677aeSAaron LI 
32411677aeSAaron LI #ifdef HAVE_CONFIG_H
33*ed775ee7SAntonio Huete Jimenez #include <config.h>
34411677aeSAaron LI #endif
35411677aeSAaron LI 
36*ed775ee7SAntonio Huete Jimenez #include "netdissect-stdinc.h"
37411677aeSAaron LI #include "netdissect.h"
38411677aeSAaron LI #include <limits.h>
39411677aeSAaron LI 
40411677aeSAaron LI #include "extract.h"
41411677aeSAaron LI 
42411677aeSAaron LI 
43411677aeSAaron LI /*
44*ed775ee7SAntonio Huete Jimenez  * For information regarding RESP, see: https://redis.io/topics/protocol
45411677aeSAaron LI  */
46411677aeSAaron LI 
47411677aeSAaron LI #define RESP_SIMPLE_STRING    '+'
48411677aeSAaron LI #define RESP_ERROR            '-'
49411677aeSAaron LI #define RESP_INTEGER          ':'
50411677aeSAaron LI #define RESP_BULK_STRING      '$'
51411677aeSAaron LI #define RESP_ARRAY            '*'
52411677aeSAaron LI 
53*ed775ee7SAntonio Huete Jimenez #define resp_print_empty(ndo)            ND_PRINT(" empty")
54*ed775ee7SAntonio Huete Jimenez #define resp_print_null(ndo)             ND_PRINT(" null")
55*ed775ee7SAntonio Huete Jimenez #define resp_print_length_too_large(ndo) ND_PRINT(" length too large")
56*ed775ee7SAntonio Huete Jimenez #define resp_print_length_negative(ndo)  ND_PRINT(" length negative and not -1")
57*ed775ee7SAntonio Huete Jimenez #define resp_print_invalid(ndo)          ND_PRINT(" invalid")
58411677aeSAaron LI 
59*ed775ee7SAntonio Huete Jimenez static int resp_parse(netdissect_options *, const u_char *, int);
60*ed775ee7SAntonio Huete Jimenez static int resp_print_string_error_integer(netdissect_options *, const u_char *, int);
61*ed775ee7SAntonio Huete Jimenez static int resp_print_simple_string(netdissect_options *, const u_char *, int);
62*ed775ee7SAntonio Huete Jimenez static int resp_print_integer(netdissect_options *, const u_char *, int);
63*ed775ee7SAntonio Huete Jimenez static int resp_print_error(netdissect_options *, const u_char *, int);
64*ed775ee7SAntonio Huete Jimenez static int resp_print_bulk_string(netdissect_options *, const u_char *, int);
65*ed775ee7SAntonio Huete Jimenez static int resp_print_bulk_array(netdissect_options *, const u_char *, int);
66*ed775ee7SAntonio Huete Jimenez static int resp_print_inline(netdissect_options *, const u_char *, int);
67*ed775ee7SAntonio Huete Jimenez static int resp_get_length(netdissect_options *, const u_char *, int, const u_char **);
68411677aeSAaron LI 
69411677aeSAaron LI #define LCHECK2(_tot_len, _len) \
70411677aeSAaron LI     {                           \
71411677aeSAaron LI         if (_tot_len < _len)    \
72411677aeSAaron LI             goto trunc;         \
73411677aeSAaron LI     }
74411677aeSAaron LI 
75411677aeSAaron LI #define LCHECK(_tot_len) LCHECK2(_tot_len, 1)
76411677aeSAaron LI 
77411677aeSAaron LI /*
78411677aeSAaron LI  * FIND_CRLF:
79411677aeSAaron LI  * Attempts to move our 'ptr' forward until a \r\n is found,
80411677aeSAaron LI  * while also making sure we don't exceed the buffer '_len'
81411677aeSAaron LI  * or go past the end of the captured data.
82411677aeSAaron LI  * If we exceed or go past the end of the captured data,
83411677aeSAaron LI  * jump to trunc.
84411677aeSAaron LI  */
85411677aeSAaron LI #define FIND_CRLF(_ptr, _len)                   \
86411677aeSAaron LI     for (;;) {                                  \
87411677aeSAaron LI         LCHECK2(_len, 2);                       \
88*ed775ee7SAntonio Huete Jimenez         ND_TCHECK_2(_ptr);                      \
89*ed775ee7SAntonio Huete Jimenez         if (GET_U_1(_ptr) == '\r' &&            \
90*ed775ee7SAntonio Huete Jimenez             GET_U_1(_ptr+1) == '\n')            \
91411677aeSAaron LI             break;                              \
92411677aeSAaron LI         _ptr++;                                 \
93411677aeSAaron LI         _len--;                                 \
94411677aeSAaron LI     }
95411677aeSAaron LI 
96411677aeSAaron LI /*
97411677aeSAaron LI  * CONSUME_CRLF
98411677aeSAaron LI  * Consume a CRLF that we've just found.
99411677aeSAaron LI  */
100411677aeSAaron LI #define CONSUME_CRLF(_ptr, _len) \
101411677aeSAaron LI     _ptr += 2;                   \
102411677aeSAaron LI     _len -= 2;
103411677aeSAaron LI 
104411677aeSAaron LI /*
105411677aeSAaron LI  * FIND_CR_OR_LF
106411677aeSAaron LI  * Attempts to move our '_ptr' forward until a \r or \n is found,
107411677aeSAaron LI  * while also making sure we don't exceed the buffer '_len'
108411677aeSAaron LI  * or go past the end of the captured data.
109411677aeSAaron LI  * If we exceed or go past the end of the captured data,
110411677aeSAaron LI  * jump to trunc.
111411677aeSAaron LI  */
112411677aeSAaron LI #define FIND_CR_OR_LF(_ptr, _len)           \
113411677aeSAaron LI     for (;;) {                              \
114411677aeSAaron LI         LCHECK(_len);                       \
115*ed775ee7SAntonio Huete Jimenez         if (GET_U_1(_ptr) == '\r' ||        \
116*ed775ee7SAntonio Huete Jimenez             GET_U_1(_ptr) == '\n')          \
117411677aeSAaron LI             break;                          \
118411677aeSAaron LI         _ptr++;                             \
119411677aeSAaron LI         _len--;                             \
120411677aeSAaron LI     }
121411677aeSAaron LI 
122411677aeSAaron LI /*
123411677aeSAaron LI  * CONSUME_CR_OR_LF
124411677aeSAaron LI  * Consume all consecutive \r and \n bytes.
125411677aeSAaron LI  * If we exceed '_len' or go past the end of the captured data,
126411677aeSAaron LI  * jump to trunc.
127411677aeSAaron LI  */
128411677aeSAaron LI #define CONSUME_CR_OR_LF(_ptr, _len)             \
129411677aeSAaron LI     {                                            \
130411677aeSAaron LI         int _found_cr_or_lf = 0;                 \
131411677aeSAaron LI         for (;;) {                               \
132411677aeSAaron LI             /*                                   \
133411677aeSAaron LI              * Have we hit the end of data?      \
134411677aeSAaron LI              */                                  \
135*ed775ee7SAntonio Huete Jimenez             if (_len == 0 || !ND_TTEST_1(_ptr)) {\
136411677aeSAaron LI                 /*                               \
137411677aeSAaron LI                  * Yes.  Have we seen a \r       \
138411677aeSAaron LI                  * or \n?                        \
139411677aeSAaron LI                  */                              \
140411677aeSAaron LI                 if (_found_cr_or_lf) {           \
141411677aeSAaron LI                     /*                           \
142411677aeSAaron LI                      * Yes.  Just stop.          \
143411677aeSAaron LI                      */                          \
144411677aeSAaron LI                     break;                       \
145411677aeSAaron LI                 }                                \
146411677aeSAaron LI                 /*                               \
147411677aeSAaron LI                  * No.  We ran out of packet.    \
148411677aeSAaron LI                  */                              \
149411677aeSAaron LI                 goto trunc;                      \
150411677aeSAaron LI             }                                    \
151*ed775ee7SAntonio Huete Jimenez             if (GET_U_1(_ptr) != '\r' &&         \
152*ed775ee7SAntonio Huete Jimenez                 GET_U_1(_ptr) != '\n')           \
153411677aeSAaron LI                 break;                           \
154411677aeSAaron LI             _found_cr_or_lf = 1;                 \
155411677aeSAaron LI             _ptr++;                              \
156411677aeSAaron LI             _len--;                              \
157411677aeSAaron LI         }                                        \
158411677aeSAaron LI     }
159411677aeSAaron LI 
160411677aeSAaron LI /*
161411677aeSAaron LI  * SKIP_OPCODE
162411677aeSAaron LI  * Skip over the opcode character.
163411677aeSAaron LI  * The opcode has already been fetched, so we know it's there, and don't
164411677aeSAaron LI  * need to do any checks.
165411677aeSAaron LI  */
166411677aeSAaron LI #define SKIP_OPCODE(_ptr, _tot_len) \
167411677aeSAaron LI     _ptr++;                         \
168411677aeSAaron LI     _tot_len--;
169411677aeSAaron LI 
170411677aeSAaron LI /*
171411677aeSAaron LI  * GET_LENGTH
172411677aeSAaron LI  * Get a bulk string or array length.
173411677aeSAaron LI  */
174411677aeSAaron LI #define GET_LENGTH(_ndo, _tot_len, _ptr, _len)                \
175411677aeSAaron LI     {                                                         \
176411677aeSAaron LI         const u_char *_endp;                                  \
177411677aeSAaron LI         _len = resp_get_length(_ndo, _ptr, _tot_len, &_endp); \
178411677aeSAaron LI         _tot_len -= (_endp - _ptr);                           \
179411677aeSAaron LI         _ptr = _endp;                                         \
180411677aeSAaron LI     }
181411677aeSAaron LI 
182411677aeSAaron LI /*
183411677aeSAaron LI  * TEST_RET_LEN
184411677aeSAaron LI  * If ret_len is < 0, jump to the trunc tag which returns (-1)
185411677aeSAaron LI  * and 'bubbles up' to printing tstr. Otherwise, return ret_len.
186411677aeSAaron LI  */
187411677aeSAaron LI #define TEST_RET_LEN(rl) \
188411677aeSAaron LI     if (rl < 0) { goto trunc; } else { return rl; }
189411677aeSAaron LI 
190411677aeSAaron LI /*
191411677aeSAaron LI  * TEST_RET_LEN_NORETURN
192411677aeSAaron LI  * If ret_len is < 0, jump to the trunc tag which returns (-1)
193411677aeSAaron LI  * and 'bubbles up' to printing tstr. Otherwise, continue onward.
194411677aeSAaron LI  */
195411677aeSAaron LI #define TEST_RET_LEN_NORETURN(rl) \
196411677aeSAaron LI     if (rl < 0) { goto trunc; }
197411677aeSAaron LI 
198411677aeSAaron LI /*
199411677aeSAaron LI  * RESP_PRINT_SEGMENT
200411677aeSAaron LI  * Prints a segment in the form of: ' "<stuff>"\n"
201411677aeSAaron LI  * Assumes the data has already been verified as present.
202411677aeSAaron LI  */
203411677aeSAaron LI #define RESP_PRINT_SEGMENT(_ndo, _bp, _len)            \
204*ed775ee7SAntonio Huete Jimenez     ND_PRINT(" \"");                                   \
205*ed775ee7SAntonio Huete Jimenez     if (nd_printn(_ndo, _bp, _len, _ndo->ndo_snapend)) \
206411677aeSAaron LI         goto trunc;                                    \
207411677aeSAaron LI     fn_print_char(_ndo, '"');
208411677aeSAaron LI 
209411677aeSAaron LI void
resp_print(netdissect_options * ndo,const u_char * bp,u_int length)210411677aeSAaron LI resp_print(netdissect_options *ndo, const u_char *bp, u_int length)
211411677aeSAaron LI {
212411677aeSAaron LI     int ret_len = 0, length_cur = length;
213411677aeSAaron LI 
214*ed775ee7SAntonio Huete Jimenez     ndo->ndo_protocol = "resp";
215411677aeSAaron LI     if(!bp || length <= 0)
216411677aeSAaron LI         return;
217411677aeSAaron LI 
218*ed775ee7SAntonio Huete Jimenez     ND_PRINT(": RESP");
219411677aeSAaron LI     while (length_cur > 0) {
220411677aeSAaron LI         /*
221411677aeSAaron LI          * This block supports redis pipelining.
222411677aeSAaron LI          * For example, multiple operations can be pipelined within the same string:
223411677aeSAaron LI          * "*2\r\n\$4\r\nINCR\r\n\$1\r\nz\r\n*2\r\n\$4\r\nINCR\r\n\$1\r\nz\r\n*2\r\n\$4\r\nINCR\r\n\$1\r\nz\r\n"
224411677aeSAaron LI          * or
225411677aeSAaron LI          * "PING\r\nPING\r\nPING\r\n"
226411677aeSAaron LI          * In order to handle this case, we must try and parse 'bp' until
227411677aeSAaron LI          * 'length' bytes have been processed or we reach a trunc condition.
228411677aeSAaron LI          */
229411677aeSAaron LI         ret_len = resp_parse(ndo, bp, length_cur);
230411677aeSAaron LI         TEST_RET_LEN_NORETURN(ret_len);
231411677aeSAaron LI         bp += ret_len;
232411677aeSAaron LI         length_cur -= ret_len;
233411677aeSAaron LI     }
234411677aeSAaron LI 
235411677aeSAaron LI     return;
236411677aeSAaron LI 
237411677aeSAaron LI trunc:
238*ed775ee7SAntonio Huete Jimenez     nd_print_trunc(ndo);
239411677aeSAaron LI }
240411677aeSAaron LI 
241411677aeSAaron LI static int
resp_parse(netdissect_options * ndo,const u_char * bp,int length)242*ed775ee7SAntonio Huete Jimenez resp_parse(netdissect_options *ndo, const u_char *bp, int length)
243411677aeSAaron LI {
244411677aeSAaron LI     u_char op;
245411677aeSAaron LI     int ret_len;
246411677aeSAaron LI 
247411677aeSAaron LI     LCHECK2(length, 1);
248*ed775ee7SAntonio Huete Jimenez     op = GET_U_1(bp);
249411677aeSAaron LI 
250411677aeSAaron LI     /* bp now points to the op, so these routines must skip it */
251411677aeSAaron LI     switch(op) {
252411677aeSAaron LI         case RESP_SIMPLE_STRING:  ret_len = resp_print_simple_string(ndo, bp, length);   break;
253411677aeSAaron LI         case RESP_INTEGER:        ret_len = resp_print_integer(ndo, bp, length);         break;
254411677aeSAaron LI         case RESP_ERROR:          ret_len = resp_print_error(ndo, bp, length);           break;
255411677aeSAaron LI         case RESP_BULK_STRING:    ret_len = resp_print_bulk_string(ndo, bp, length);     break;
256411677aeSAaron LI         case RESP_ARRAY:          ret_len = resp_print_bulk_array(ndo, bp, length);      break;
257411677aeSAaron LI         default:                  ret_len = resp_print_inline(ndo, bp, length);          break;
258411677aeSAaron LI     }
259411677aeSAaron LI 
260411677aeSAaron LI     /*
261411677aeSAaron LI      * This gives up with a "truncated" indicator for all errors,
262411677aeSAaron LI      * including invalid packet errors; that's what we want, as
263411677aeSAaron LI      * we have to give up on further parsing in that case.
264411677aeSAaron LI      */
265411677aeSAaron LI     TEST_RET_LEN(ret_len);
266411677aeSAaron LI 
267411677aeSAaron LI trunc:
268411677aeSAaron LI     return (-1);
269411677aeSAaron LI }
270411677aeSAaron LI 
271411677aeSAaron LI static int
resp_print_simple_string(netdissect_options * ndo,const u_char * bp,int length)272*ed775ee7SAntonio Huete Jimenez resp_print_simple_string(netdissect_options *ndo, const u_char *bp, int length) {
273411677aeSAaron LI     return resp_print_string_error_integer(ndo, bp, length);
274411677aeSAaron LI }
275411677aeSAaron LI 
276411677aeSAaron LI static int
resp_print_integer(netdissect_options * ndo,const u_char * bp,int length)277*ed775ee7SAntonio Huete Jimenez resp_print_integer(netdissect_options *ndo, const u_char *bp, int length) {
278411677aeSAaron LI     return resp_print_string_error_integer(ndo, bp, length);
279411677aeSAaron LI }
280411677aeSAaron LI 
281411677aeSAaron LI static int
resp_print_error(netdissect_options * ndo,const u_char * bp,int length)282*ed775ee7SAntonio Huete Jimenez resp_print_error(netdissect_options *ndo, const u_char *bp, int length) {
283411677aeSAaron LI     return resp_print_string_error_integer(ndo, bp, length);
284411677aeSAaron LI }
285411677aeSAaron LI 
286411677aeSAaron LI static int
resp_print_string_error_integer(netdissect_options * ndo,const u_char * bp,int length)287*ed775ee7SAntonio Huete Jimenez resp_print_string_error_integer(netdissect_options *ndo, const u_char *bp, int length) {
288411677aeSAaron LI     int length_cur = length, len, ret_len;
289411677aeSAaron LI     const u_char *bp_ptr;
290411677aeSAaron LI 
291411677aeSAaron LI     /* bp points to the op; skip it */
292411677aeSAaron LI     SKIP_OPCODE(bp, length_cur);
293411677aeSAaron LI     bp_ptr = bp;
294411677aeSAaron LI 
295411677aeSAaron LI     /*
296411677aeSAaron LI      * bp now prints past the (+-;) opcode, so it's pointing to the first
297411677aeSAaron LI      * character of the string (which could be numeric).
298411677aeSAaron LI      * +OK\r\n
299411677aeSAaron LI      * -ERR ...\r\n
300411677aeSAaron LI      * :02912309\r\n
301411677aeSAaron LI      *
302411677aeSAaron LI      * Find the \r\n with FIND_CRLF().
303411677aeSAaron LI      */
304411677aeSAaron LI     FIND_CRLF(bp_ptr, length_cur);
305411677aeSAaron LI 
306411677aeSAaron LI     /*
307411677aeSAaron LI      * bp_ptr points to the \r\n, so bp_ptr - bp is the length of text
308411677aeSAaron LI      * preceding the \r\n.  That includes the opcode, so don't print
309411677aeSAaron LI      * that.
310411677aeSAaron LI      */
311*ed775ee7SAntonio Huete Jimenez     len = ND_BYTES_BETWEEN(bp_ptr, bp);
312411677aeSAaron LI     RESP_PRINT_SEGMENT(ndo, bp, len);
313411677aeSAaron LI     ret_len = 1 /*<opcode>*/ + len /*<string>*/ + 2 /*<CRLF>*/;
314411677aeSAaron LI 
315411677aeSAaron LI     TEST_RET_LEN(ret_len);
316411677aeSAaron LI 
317411677aeSAaron LI trunc:
318411677aeSAaron LI     return (-1);
319411677aeSAaron LI }
320411677aeSAaron LI 
321411677aeSAaron LI static int
resp_print_bulk_string(netdissect_options * ndo,const u_char * bp,int length)322*ed775ee7SAntonio Huete Jimenez resp_print_bulk_string(netdissect_options *ndo, const u_char *bp, int length) {
323411677aeSAaron LI     int length_cur = length, string_len;
324411677aeSAaron LI 
325411677aeSAaron LI     /* bp points to the op; skip it */
326411677aeSAaron LI     SKIP_OPCODE(bp, length_cur);
327411677aeSAaron LI 
328411677aeSAaron LI     /* <length>\r\n */
329411677aeSAaron LI     GET_LENGTH(ndo, length_cur, bp, string_len);
330411677aeSAaron LI 
331411677aeSAaron LI     if (string_len >= 0) {
332411677aeSAaron LI         /* Byte string of length string_len, starting at bp */
333411677aeSAaron LI         if (string_len == 0)
334411677aeSAaron LI             resp_print_empty(ndo);
335411677aeSAaron LI         else {
336411677aeSAaron LI             LCHECK2(length_cur, string_len);
337*ed775ee7SAntonio Huete Jimenez             ND_TCHECK_LEN(bp, string_len);
338411677aeSAaron LI             RESP_PRINT_SEGMENT(ndo, bp, string_len);
339411677aeSAaron LI             bp += string_len;
340411677aeSAaron LI             length_cur -= string_len;
341411677aeSAaron LI         }
342411677aeSAaron LI 
343411677aeSAaron LI         /*
344411677aeSAaron LI          * Find the \r\n at the end of the string and skip past it.
345411677aeSAaron LI          * XXX - report an error if the \r\n isn't immediately after
346411677aeSAaron LI          * the item?
347411677aeSAaron LI          */
348411677aeSAaron LI         FIND_CRLF(bp, length_cur);
349411677aeSAaron LI         CONSUME_CRLF(bp, length_cur);
350411677aeSAaron LI     } else {
351411677aeSAaron LI         /* null, truncated, or invalid for some reason */
352411677aeSAaron LI         switch(string_len) {
353411677aeSAaron LI             case (-1):  resp_print_null(ndo);             break;
354411677aeSAaron LI             case (-2):  goto trunc;
355411677aeSAaron LI             case (-3):  resp_print_length_too_large(ndo); break;
356411677aeSAaron LI             case (-4):  resp_print_length_negative(ndo);  break;
357411677aeSAaron LI             default:    resp_print_invalid(ndo);          break;
358411677aeSAaron LI         }
359411677aeSAaron LI     }
360411677aeSAaron LI 
361411677aeSAaron LI     return (length - length_cur);
362411677aeSAaron LI 
363411677aeSAaron LI trunc:
364411677aeSAaron LI     return (-1);
365411677aeSAaron LI }
366411677aeSAaron LI 
367411677aeSAaron LI static int
resp_print_bulk_array(netdissect_options * ndo,const u_char * bp,int length)368*ed775ee7SAntonio Huete Jimenez resp_print_bulk_array(netdissect_options *ndo, const u_char *bp, int length) {
369411677aeSAaron LI     u_int length_cur = length;
370411677aeSAaron LI     int array_len, i, ret_len;
371411677aeSAaron LI 
372411677aeSAaron LI     /* bp points to the op; skip it */
373411677aeSAaron LI     SKIP_OPCODE(bp, length_cur);
374411677aeSAaron LI 
375411677aeSAaron LI     /* <array_length>\r\n */
376411677aeSAaron LI     GET_LENGTH(ndo, length_cur, bp, array_len);
377411677aeSAaron LI 
378411677aeSAaron LI     if (array_len > 0) {
379411677aeSAaron LI         /* non empty array */
380411677aeSAaron LI         for (i = 0; i < array_len; i++) {
381411677aeSAaron LI             ret_len = resp_parse(ndo, bp, length_cur);
382411677aeSAaron LI 
383411677aeSAaron LI             TEST_RET_LEN_NORETURN(ret_len);
384411677aeSAaron LI 
385411677aeSAaron LI             bp += ret_len;
386411677aeSAaron LI             length_cur -= ret_len;
387411677aeSAaron LI         }
388411677aeSAaron LI     } else {
389411677aeSAaron LI         /* empty, null, truncated, or invalid */
390411677aeSAaron LI         switch(array_len) {
391411677aeSAaron LI             case 0:     resp_print_empty(ndo);            break;
392411677aeSAaron LI             case (-1):  resp_print_null(ndo);             break;
393411677aeSAaron LI             case (-2):  goto trunc;
394411677aeSAaron LI             case (-3):  resp_print_length_too_large(ndo); break;
395411677aeSAaron LI             case (-4):  resp_print_length_negative(ndo);  break;
396411677aeSAaron LI             default:    resp_print_invalid(ndo);          break;
397411677aeSAaron LI         }
398411677aeSAaron LI     }
399411677aeSAaron LI 
400411677aeSAaron LI     return (length - length_cur);
401411677aeSAaron LI 
402411677aeSAaron LI trunc:
403411677aeSAaron LI     return (-1);
404411677aeSAaron LI }
405411677aeSAaron LI 
406411677aeSAaron LI static int
resp_print_inline(netdissect_options * ndo,const u_char * bp,int length)407*ed775ee7SAntonio Huete Jimenez resp_print_inline(netdissect_options *ndo, const u_char *bp, int length) {
408411677aeSAaron LI     int length_cur = length;
409411677aeSAaron LI     int len;
410411677aeSAaron LI     const u_char *bp_ptr;
411411677aeSAaron LI 
412411677aeSAaron LI     /*
413411677aeSAaron LI      * Inline commands are simply 'strings' followed by \r or \n or both.
414411677aeSAaron LI      * Redis will do its best to split/parse these strings.
415411677aeSAaron LI      * This feature of redis is implemented to support the ability of
416411677aeSAaron LI      * command parsing from telnet/nc sessions etc.
417411677aeSAaron LI      *
418411677aeSAaron LI      * <string><\r||\n||\r\n...>
419411677aeSAaron LI      */
420411677aeSAaron LI 
421411677aeSAaron LI     /*
422411677aeSAaron LI      * Skip forward past any leading \r, \n, or \r\n.
423411677aeSAaron LI      */
424411677aeSAaron LI     CONSUME_CR_OR_LF(bp, length_cur);
425411677aeSAaron LI     bp_ptr = bp;
426411677aeSAaron LI 
427411677aeSAaron LI     /*
428411677aeSAaron LI      * Scan forward looking for \r or \n.
429411677aeSAaron LI      */
430411677aeSAaron LI     FIND_CR_OR_LF(bp_ptr, length_cur);
431411677aeSAaron LI 
432411677aeSAaron LI     /*
433411677aeSAaron LI      * Found it; bp_ptr points to the \r or \n, so bp_ptr - bp is the
434*ed775ee7SAntonio Huete Jimenez      * Length of the line text that precedes it.  Print it.
435411677aeSAaron LI      */
436*ed775ee7SAntonio Huete Jimenez     len = ND_BYTES_BETWEEN(bp_ptr, bp);
437411677aeSAaron LI     RESP_PRINT_SEGMENT(ndo, bp, len);
438411677aeSAaron LI 
439411677aeSAaron LI     /*
440411677aeSAaron LI      * Skip forward past the \r, \n, or \r\n.
441411677aeSAaron LI      */
442411677aeSAaron LI     CONSUME_CR_OR_LF(bp_ptr, length_cur);
443411677aeSAaron LI 
444411677aeSAaron LI     /*
445411677aeSAaron LI      * Return the number of bytes we processed.
446411677aeSAaron LI      */
447411677aeSAaron LI     return (length - length_cur);
448411677aeSAaron LI 
449411677aeSAaron LI trunc:
450411677aeSAaron LI     return (-1);
451411677aeSAaron LI }
452411677aeSAaron LI 
453411677aeSAaron LI static int
resp_get_length(netdissect_options * ndo,const u_char * bp,int len,const u_char ** endp)454*ed775ee7SAntonio Huete Jimenez resp_get_length(netdissect_options *ndo, const u_char *bp, int len, const u_char **endp)
455411677aeSAaron LI {
456411677aeSAaron LI     int result;
457411677aeSAaron LI     u_char c;
458411677aeSAaron LI     int saw_digit;
459411677aeSAaron LI     int neg;
460411677aeSAaron LI     int too_large;
461411677aeSAaron LI 
462411677aeSAaron LI     if (len == 0)
463411677aeSAaron LI         goto trunc;
464411677aeSAaron LI     too_large = 0;
465411677aeSAaron LI     neg = 0;
466*ed775ee7SAntonio Huete Jimenez     if (GET_U_1(bp) == '-') {
467411677aeSAaron LI         neg = 1;
468411677aeSAaron LI         bp++;
469411677aeSAaron LI         len--;
470411677aeSAaron LI     }
471411677aeSAaron LI     result = 0;
472411677aeSAaron LI     saw_digit = 0;
473411677aeSAaron LI 
474411677aeSAaron LI     for (;;) {
475411677aeSAaron LI         if (len == 0)
476411677aeSAaron LI             goto trunc;
477*ed775ee7SAntonio Huete Jimenez         c = GET_U_1(bp);
478411677aeSAaron LI         if (!(c >= '0' && c <= '9')) {
479411677aeSAaron LI             if (!saw_digit) {
480411677aeSAaron LI                 bp++;
481411677aeSAaron LI                 goto invalid;
482411677aeSAaron LI             }
483411677aeSAaron LI             break;
484411677aeSAaron LI         }
485411677aeSAaron LI         c -= '0';
486411677aeSAaron LI         if (result > (INT_MAX / 10)) {
487411677aeSAaron LI             /* This will overflow an int when we multiply it by 10. */
488411677aeSAaron LI             too_large = 1;
489411677aeSAaron LI         } else {
490411677aeSAaron LI             result *= 10;
491411677aeSAaron LI             if (result == ((INT_MAX / 10) * 10) && c > (INT_MAX % 10)) {
492411677aeSAaron LI                 /* This will overflow an int when we add c */
493411677aeSAaron LI                 too_large = 1;
494411677aeSAaron LI             } else
495411677aeSAaron LI                 result += c;
496411677aeSAaron LI         }
497411677aeSAaron LI         bp++;
498411677aeSAaron LI         len--;
499411677aeSAaron LI         saw_digit = 1;
500411677aeSAaron LI     }
501411677aeSAaron LI 
502411677aeSAaron LI     /*
503411677aeSAaron LI      * OK, we found a non-digit character.  It should be a \r, followed
504411677aeSAaron LI      * by a \n.
505411677aeSAaron LI      */
506*ed775ee7SAntonio Huete Jimenez     if (GET_U_1(bp) != '\r') {
507411677aeSAaron LI         bp++;
508411677aeSAaron LI         goto invalid;
509411677aeSAaron LI     }
510411677aeSAaron LI     bp++;
511411677aeSAaron LI     len--;
512411677aeSAaron LI     if (len == 0)
513411677aeSAaron LI         goto trunc;
514*ed775ee7SAntonio Huete Jimenez     if (GET_U_1(bp) != '\n') {
515411677aeSAaron LI         bp++;
516411677aeSAaron LI         goto invalid;
517411677aeSAaron LI     }
518411677aeSAaron LI     bp++;
519411677aeSAaron LI     len--;
520411677aeSAaron LI     *endp = bp;
521411677aeSAaron LI     if (neg) {
522411677aeSAaron LI         /* -1 means "null", anything else is invalid */
523411677aeSAaron LI         if (too_large || result != 1)
524411677aeSAaron LI             return (-4);
525411677aeSAaron LI         result = -1;
526411677aeSAaron LI     }
527411677aeSAaron LI     return (too_large ? -3 : result);
528411677aeSAaron LI 
529411677aeSAaron LI trunc:
530411677aeSAaron LI     *endp = bp;
531411677aeSAaron LI     return (-2);
532411677aeSAaron LI 
533411677aeSAaron LI invalid:
534411677aeSAaron LI     *endp = bp;
535411677aeSAaron LI     return (-5);
536411677aeSAaron LI }
537