Coverage Report

Created: 2025-10-10 06:06

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/opensips/parser/parse_via.c
Line
Count
Source
1
/*
2
 * via parsing automaton
3
 *
4
 * Copyright (C) 2001-2003 FhG Fokus
5
 *
6
 * This file is part of opensips, a free SIP server.
7
 *
8
 * opensips is free software; you can redistribute it and/or modify
9
 * it under the terms of the GNU General Public License as published by
10
 * the Free Software Foundation; either version 2 of the License, or
11
 * (at your option) any later version
12
 *
13
 * opensips is distributed in the hope that it will be useful,
14
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16
 * GNU General Public License for more details.
17
 *
18
 * You should have received a copy of the GNU General Public License
19
 * along with this program; if not, write to the Free Software
20
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301  USA
21
 */
22
23
24
25
/*
26
 *  2003-01-21  added rport parsing code, contributed by
27
 *               Maxim Sobolev  <sobomax@FreeBSD.org>
28
 *  2003-01-23  added extra via param parsing code (i=...), used
29
 *               by tcp to identify the sending socket, by andrei
30
 *  2003-01-23  fixed rport parsing code to accept rport w/o any value,
31
 *               by andrei
32
 *  2003-01-27  modified parse_via to set new via_param->start member and
33
 *               via->params.s (andrei)
34
 *  2003-01-28  zero-terminations replaced with VIA_ZT (jiri)
35
 *  2003-02-28  scratchpad compatibility abandoned (jiri)
36
 *  2003-04-26  ZSW (jiri)
37
 *  2003-06-23  fixed  parse_via_param [op].* param. parsing bug (andrei)
38
 *  2003-07-02  added support for TLS parsing in via (andrei)
39
 *  2003-10-27  added support for alias via param parsing [see
40
 *               draft-ietf-sip-connect-reuse-00.txt.]  (andrei)
41
 *  2004-03-31  fixed rport set instead of i bug (andrei)
42
 *  2005-03-02  if via has multiple bodies, and one of them is bad set
43
 *               also the first one as bad (andrei)
44
 */
45
46
47
48
#include <stdlib.h>
49
#include <string.h>
50
#include "../dprint.h"
51
#include "../ut.h"
52
#include "../ip_addr.h"
53
#include "../mem/mem.h"
54
#include "parse_via.h"
55
#include "parse_def.h"
56
57
58
59
/* main via states (uri:port ...) */
60
enum {
61
  F_HOST, P_HOST,
62
  L_PORT, F_PORT, P_PORT,
63
  L_PARAM, F_PARAM, P_PARAM,
64
  L_VIA, F_VIA,
65
  F_COMMENT, P_COMMENT,
66
  F_IP6HOST, P_IP6HOST,
67
  F_CRLF,
68
  F_LF,
69
  F_CR,
70
  END_OF_HEADER
71
};
72
73
74
/* first via part state */
75
enum {
76
  F_SIP = 100,
77
  SIP1, SIP2, FIN_SIP,
78
  L_VER, F_VER,
79
  VER1, VER2, FIN_VER,
80
  UDP1, UDP2, FIN_UDP,
81
  TCP_TLS1, TCP2, FIN_TCP,
82
  TLS2, FIN_TLS,
83
  SCTP1, SCTP2, SCTP3, FIN_SCTP,
84
  WS1, WS_WSS, FIN_WS, FIN_WSS,
85
  OTHER_PROTO,
86
  L_PROTO, F_PROTO
87
};
88
89
90
/* param related states
91
 * WARNING: keep in sync with parse_via.h, PARAM_HIDDEN, ...
92
 */
93
enum {
94
  L_VALUE = 200, F_VALUE, P_VALUE, P_STRING,
95
  HIDDEN1, HIDDEN2, HIDDEN3, HIDDEN4, HIDDEN5,
96
  TTL1, TTL2,
97
  BRANCH1, BRANCH2, BRANCH3, BRANCH4, BRANCH5,
98
  MADDR1, MADDR2, MADDR3, MADDR4,
99
  RECEIVED1, RECEIVED2, RECEIVED3, RECEIVED4, RECEIVED5, RECEIVED6,
100
  RECEIVED7,
101
  RPORT1, RPORT2, RPORT3,
102
  ALIAS1, ALIAS2, ALIAS3, ALIAS4,
103
       /* fin states (227-...)*/
104
  FIN_HIDDEN = 230, FIN_TTL, FIN_BRANCH,
105
  FIN_MADDR, FIN_RECEIVED, FIN_RPORT, FIN_I, FIN_ALIAS
106
       /*GEN_PARAM,
107
         PARAM_ERROR*/ /* declared in msg_parser.h*/
108
};
109
110
111
/* entry state must be F_PARAM, or saved_state=F_PARAM and
112
 * state=F_{LF,CR,CRLF}!
113
 * output state = L_PARAM or F_PARAM or END_OF_HEADER
114
 * (and saved_state= last state); everything else => error
115
 * WARNING: param->start must be filled before, it's used in param->size
116
 * computation.
117
 */
118
static /*inline*/ char* parse_via_param(char* p, char* end,
119
                    unsigned char* pstate,
120
                      unsigned char* psaved_state,
121
                    struct via_param* param)
122
0
{
123
0
  char* tmp;
124
0
  register unsigned char state;
125
0
  unsigned char saved_state;
126
127
0
  state=*pstate;
128
129
0
  saved_state=*psaved_state;
130
0
  param->type=PARAM_ERROR;
131
132
0
  for (tmp=p;tmp<end;tmp++){
133
0
    switch(*tmp){
134
0
      case ' ':
135
0
      case '\t':
136
0
        switch(state){
137
0
          case FIN_HIDDEN:
138
0
          case FIN_ALIAS:
139
0
            param->type=state;
140
0
            param->name.len=tmp-param->name.s;
141
0
            state=L_PARAM;
142
0
            goto endofparam;
143
0
          case FIN_BRANCH:
144
0
          case FIN_TTL:
145
0
          case FIN_MADDR:
146
0
          case FIN_RECEIVED:
147
0
          case FIN_RPORT:
148
0
          case FIN_I:
149
0
            param->type=state;
150
0
            param->name.len=tmp-param->name.s;
151
0
            state=L_VALUE;
152
0
            goto find_value;
153
0
          case F_PARAM:
154
0
            break;
155
0
          case F_LF:
156
0
          case F_CR:
157
0
          case F_CRLF:
158
0
            state=saved_state;
159
0
            break;
160
0
          case GEN_PARAM:
161
0
          default:
162
0
            param->type=GEN_PARAM;
163
0
            param->name.len=tmp-param->name.s;
164
0
            state=L_VALUE;
165
0
            goto find_value;
166
0
        }
167
0
        break;
168
      /* \n and \r*/
169
0
      case '\n':
170
0
        switch(state){
171
0
          case FIN_HIDDEN:
172
0
          case FIN_ALIAS:
173
0
            param->type=state;
174
0
            param->name.len=tmp-param->name.s;
175
0
            param->size=tmp-param->start;
176
0
            saved_state=L_PARAM;
177
0
            state=F_LF;
178
0
            goto endofparam;
179
0
          case FIN_BRANCH:
180
0
          case FIN_TTL:
181
0
          case FIN_MADDR:
182
0
          case FIN_RECEIVED:
183
0
          case FIN_I:
184
0
          case FIN_RPORT:
185
0
            param->type=state;
186
0
            param->name.len=tmp-param->name.s;
187
0
            param->size=tmp-param->start;
188
0
            saved_state=L_VALUE;
189
0
            state=F_LF;
190
0
            goto find_value;
191
0
          case F_PARAM:
192
0
            saved_state=state;
193
0
            state=F_LF;
194
0
            break;
195
0
          case F_LF:
196
0
          case F_CRLF:
197
0
            state=END_OF_HEADER;
198
0
            goto end_via;
199
0
          case F_CR:
200
0
            state=F_CRLF;
201
0
            break;
202
0
          case GEN_PARAM:
203
0
          default:
204
0
            param->type=GEN_PARAM;
205
0
            saved_state=L_VALUE;
206
0
            param->name.len=tmp-param->name.s;
207
0
            param->size=tmp-param->start;
208
0
            state=F_LF;
209
0
            goto find_value;
210
0
        }
211
0
        break;
212
0
      case '\r':
213
0
        switch(state){
214
0
          case FIN_HIDDEN:
215
0
          case FIN_ALIAS:
216
0
            param->type=state;
217
0
            param->name.len=tmp-param->name.s;
218
0
            param->size=tmp-param->start;
219
0
            saved_state=L_PARAM;
220
0
            state=F_CR;
221
0
            goto endofparam;
222
0
          case FIN_BRANCH:
223
0
          case FIN_TTL:
224
0
          case FIN_MADDR:
225
0
          case FIN_RECEIVED:
226
0
          case FIN_I:
227
0
          case FIN_RPORT:
228
0
            param->type=state;
229
0
            param->name.len=tmp-param->name.s;
230
0
            param->size=tmp-param->start;
231
0
            saved_state=L_VALUE;
232
0
            state=F_CR;
233
0
            goto find_value;
234
0
          case F_PARAM:
235
0
            saved_state=state;
236
0
            state=F_CR;
237
0
            break;
238
0
          case F_CR:
239
0
          case F_CRLF:
240
0
            state=END_OF_HEADER;
241
0
            goto end_via;
242
0
          case GEN_PARAM:
243
0
          default:
244
0
            param->type=GEN_PARAM;
245
0
            param->name.len=tmp-param->name.s;
246
0
            param->size=tmp-param->start;
247
0
            saved_state=L_VALUE;
248
0
            state=F_CR;
249
0
            goto find_value;
250
0
        }
251
0
        break;
252
253
0
      case '=':
254
0
        switch(state){
255
0
          case FIN_BRANCH:
256
0
          case FIN_TTL:
257
0
          case FIN_MADDR:
258
0
          case FIN_RECEIVED:
259
0
          case FIN_RPORT:
260
0
          case FIN_I:
261
0
            param->type=state;
262
0
            param->name.len=tmp-param->name.s;
263
0
            state=F_VALUE;
264
0
            goto find_value;
265
0
          case F_PARAM:
266
0
          case FIN_HIDDEN:
267
0
          case FIN_ALIAS:
268
0
            LM_ERR("invalid char <%c> in state %d\n", *tmp, state);
269
0
            goto parse_error;
270
0
          case F_CR:
271
0
          case F_LF:
272
0
          case F_CRLF:
273
0
            state=END_OF_HEADER;
274
0
            goto end_via;
275
0
          case GEN_PARAM:
276
0
          default:
277
0
            param->type=GEN_PARAM;
278
0
            param->name.len=tmp-param->name.s;
279
0
            state=F_VALUE;
280
0
            goto find_value;
281
0
        }
282
0
        break;
283
0
      case ';':
284
0
        switch(state){
285
0
          case FIN_HIDDEN:
286
0
          case FIN_RPORT: /* rport can appear w/o a value */
287
0
          case FIN_ALIAS:
288
0
            param->type=state;
289
0
            param->name.len=tmp-param->name.s;
290
0
            state=F_PARAM;
291
0
            goto endofparam;
292
0
          case FIN_BRANCH:
293
0
          case FIN_MADDR:
294
0
          case FIN_TTL:
295
0
          case FIN_RECEIVED:
296
0
          case FIN_I:
297
0
            LM_ERR("invalid char <%c> in state %d\n", *tmp, state);
298
0
            goto parse_error;
299
0
          case F_CR:
300
0
          case F_LF:
301
0
          case F_CRLF:
302
0
            state=END_OF_HEADER;
303
0
            goto end_via;
304
0
          case GEN_PARAM:
305
0
          default:
306
0
            param->type=GEN_PARAM;
307
0
            param->name.len=tmp-param->name.s;
308
0
            state=F_PARAM;
309
0
            goto endofparam;
310
0
        }
311
0
        break;
312
0
      case ',':
313
0
        switch(state){
314
0
          case FIN_HIDDEN:
315
0
          case FIN_RPORT:
316
0
          case FIN_ALIAS:
317
0
            param->type=state;
318
0
            param->name.len=tmp-param->name.s;
319
0
            state=F_VIA;
320
0
            goto endofvalue;
321
0
          case FIN_BRANCH:
322
0
          case FIN_MADDR:
323
0
          case FIN_TTL:
324
0
          case FIN_RECEIVED:
325
0
          case FIN_I:
326
0
            LM_ERR("new via found (',') when '=' expected"
327
0
                "(state %d=)\n", state);
328
0
            goto parse_error; /* or we could ignore this bad param*/
329
0
          case F_CR:
330
0
          case F_LF:
331
0
          case F_CRLF:
332
0
            state=END_OF_HEADER;
333
0
            goto end_via;
334
0
          case GEN_PARAM:
335
0
          default:
336
0
            param->type=GEN_PARAM;
337
0
            param->name.len=tmp-param->name.s;
338
0
            state=F_VIA;
339
0
            goto endofvalue;
340
0
        }
341
0
        break;
342
343
        /* param names */
344
0
      case 'h':
345
0
      case 'H':
346
0
        switch(state){
347
0
          case F_PARAM:
348
0
            state=HIDDEN1;
349
0
            param->name.s=tmp;
350
0
            break;
351
0
          case BRANCH5:
352
0
            state=FIN_BRANCH;
353
0
            break;
354
0
          case GEN_PARAM:
355
0
            break;
356
0
          case F_CR:
357
0
          case F_LF:
358
0
          case F_CRLF:
359
0
            state=END_OF_HEADER;
360
0
            goto end_via;
361
0
          default:
362
0
            state=GEN_PARAM;
363
0
        }
364
0
        break;
365
0
      case 'i':
366
0
      case 'I':
367
0
        switch(state){
368
0
          case F_PARAM:
369
0
            state=FIN_I;
370
0
            param->name.s=tmp;
371
0
            break;
372
0
          case HIDDEN1:
373
0
            state=HIDDEN2;
374
0
            break;
375
0
          case RECEIVED4:
376
0
            state=RECEIVED5;
377
0
            break;
378
0
          case ALIAS2:
379
0
            state=ALIAS3;
380
0
            break;
381
0
          case GEN_PARAM:
382
0
            break;
383
0
          case F_CR:
384
0
          case F_LF:
385
0
          case F_CRLF:
386
0
            state=END_OF_HEADER;
387
0
            goto end_via;
388
0
          default:
389
0
            state=GEN_PARAM;
390
0
        }
391
0
        break;
392
0
      case 'd':
393
0
      case 'D':
394
0
        switch(state){
395
0
          case F_PARAM:
396
0
            state=GEN_PARAM;
397
0
            param->name.s=tmp;
398
0
            break;
399
0
          case HIDDEN2:
400
0
            state=HIDDEN3;
401
0
            break;
402
0
          case HIDDEN3:
403
0
            state=HIDDEN4;
404
0
            break;
405
0
          case MADDR2:
406
0
            state=MADDR3;
407
0
            break;
408
0
          case MADDR3:
409
0
            state=MADDR4;
410
0
            break;
411
0
          case RECEIVED7:
412
0
            state=FIN_RECEIVED;
413
0
            break;
414
0
          case GEN_PARAM:
415
0
            break;
416
0
          case F_CR:
417
0
          case F_LF:
418
0
          case F_CRLF:
419
0
            state=END_OF_HEADER;
420
0
            goto end_via;
421
0
          default:
422
0
            state=GEN_PARAM;
423
0
        }
424
0
        break;
425
0
      case 'e':
426
0
      case 'E':
427
0
        switch(state){
428
0
          case F_PARAM:
429
0
            state=GEN_PARAM;
430
0
            param->name.s=tmp;
431
0
            break;
432
0
          case HIDDEN4:
433
0
            state=HIDDEN5;
434
0
            break;
435
0
          case RECEIVED1:
436
0
            state=RECEIVED2;
437
0
            break;
438
0
          case RECEIVED3:
439
0
            state=RECEIVED4;
440
0
            break;
441
0
          case RECEIVED6:
442
0
            state=RECEIVED7;
443
0
            break;
444
0
          case GEN_PARAM:
445
0
            break;
446
0
          case F_CR:
447
0
          case F_LF:
448
0
          case F_CRLF:
449
0
            state=END_OF_HEADER;
450
0
            goto end_via;
451
0
          default:
452
0
            state=GEN_PARAM;
453
0
        }
454
0
        break;
455
0
      case 'n':
456
0
      case 'N':
457
0
        switch(state){
458
0
          case F_PARAM:
459
0
            state=GEN_PARAM;
460
0
            param->name.s=tmp;
461
0
            break;
462
0
          case HIDDEN5:
463
0
            state=FIN_HIDDEN;
464
0
            break;
465
0
          case BRANCH3:
466
0
            state=BRANCH4;
467
0
            break;
468
0
          case GEN_PARAM:
469
0
            break;
470
0
          case F_CR:
471
0
          case F_LF:
472
0
          case F_CRLF:
473
0
            state=END_OF_HEADER;
474
0
            goto end_via;
475
0
          default:
476
0
            state=GEN_PARAM;
477
0
        }
478
0
        break;
479
0
      case 't':
480
0
      case 'T':
481
0
        switch(state){
482
0
          case F_PARAM:
483
0
            state=TTL1;
484
0
            param->name.s=tmp;
485
0
            break;
486
0
          case TTL1:
487
0
            state=TTL2;
488
0
            break;
489
0
          case RPORT3:
490
0
            state=FIN_RPORT;
491
0
            break;
492
0
          case GEN_PARAM:
493
0
            break;
494
0
          case F_CR:
495
0
          case F_LF:
496
0
          case F_CRLF:
497
0
            state=END_OF_HEADER;
498
0
            goto end_via;
499
0
          default:
500
0
            state=GEN_PARAM;
501
0
        }
502
0
        break;
503
0
      case 'l':
504
0
      case 'L':
505
0
        switch(state){
506
0
          case F_PARAM:
507
0
            state=GEN_PARAM;
508
0
            param->name.s=tmp;
509
0
            break;
510
0
          case TTL2:
511
0
            state=FIN_TTL;
512
0
            break;
513
0
          case ALIAS1:
514
0
            state=ALIAS2;
515
0
            break;
516
0
          case GEN_PARAM:
517
0
            break;
518
0
          case F_CR:
519
0
          case F_LF:
520
0
          case F_CRLF:
521
0
            state=END_OF_HEADER;
522
0
            goto end_via;
523
0
          default:
524
0
            state=GEN_PARAM;
525
0
        }
526
0
        break;
527
0
      case 'm':
528
0
      case 'M':
529
0
        switch(state){
530
0
          case F_PARAM:
531
0
            state=MADDR1;
532
0
            param->name.s=tmp;
533
0
            break;
534
0
          case GEN_PARAM:
535
0
            break;
536
0
          case F_CR:
537
0
          case F_LF:
538
0
          case F_CRLF:
539
0
            state=END_OF_HEADER;
540
0
            goto end_via;
541
0
          default:
542
0
            state=GEN_PARAM;
543
0
        }
544
0
        break;
545
0
      case 'a':
546
0
      case 'A':
547
0
        switch(state){
548
0
          case F_PARAM:
549
0
            state=ALIAS1;
550
0
            param->name.s=tmp;
551
0
            break;
552
0
          case MADDR1:
553
0
            state=MADDR2;
554
0
            break;
555
0
          case BRANCH2:
556
0
            state=BRANCH3;
557
0
            break;
558
0
          case ALIAS3:
559
0
            state=ALIAS4;
560
0
            break;
561
0
          case GEN_PARAM:
562
0
            break;
563
0
          case F_CR:
564
0
          case F_LF:
565
0
          case F_CRLF:
566
0
            state=END_OF_HEADER;
567
0
            goto end_via;
568
0
          default:
569
0
            state=GEN_PARAM;
570
0
        }
571
0
        break;
572
0
      case 'r':
573
0
      case 'R':
574
0
        switch(state){
575
0
          case MADDR4:
576
0
            state=FIN_MADDR;
577
0
            break;
578
0
          case F_PARAM:
579
0
            state=RECEIVED1;
580
0
            param->name.s=tmp;
581
0
            break;
582
0
          case BRANCH1:
583
0
            state=BRANCH2;
584
0
            break;
585
0
          case RPORT2:
586
0
            state=RPORT3;
587
0
            break;
588
0
          case GEN_PARAM:
589
0
            break;
590
0
          case F_CR:
591
0
          case F_LF:
592
0
          case F_CRLF:
593
0
            state=END_OF_HEADER;
594
0
            goto end_via;
595
0
          default:
596
0
            state=GEN_PARAM;
597
0
        }
598
0
        break;
599
0
      case 'c':
600
0
      case 'C':
601
0
        switch(state){
602
0
          case F_PARAM:
603
0
            state=GEN_PARAM;
604
0
            param->name.s=tmp;
605
0
            break;
606
0
          case RECEIVED2:
607
0
            state=RECEIVED3;
608
0
            break;
609
0
          case BRANCH4:
610
0
            state=BRANCH5;
611
0
            break;
612
0
          case GEN_PARAM:
613
0
            break;
614
0
          case F_CR:
615
0
          case F_LF:
616
0
          case F_CRLF:
617
0
            state=END_OF_HEADER;
618
0
            goto end_via;
619
0
          default:
620
0
            state=GEN_PARAM;
621
0
        }
622
0
        break;
623
0
      case 'v':
624
0
      case 'V':
625
0
        switch(state){
626
0
          case F_PARAM:
627
0
            state=GEN_PARAM;
628
0
            param->name.s=tmp;
629
0
            break;
630
0
          case RECEIVED5:
631
0
            state=RECEIVED6;
632
0
            break;
633
0
          case GEN_PARAM:
634
0
            break;
635
0
          case F_CR:
636
0
          case F_LF:
637
0
          case F_CRLF:
638
0
            state=END_OF_HEADER;
639
0
            goto end_via;
640
0
          default:
641
0
            state=GEN_PARAM;
642
0
        }
643
0
        break;
644
0
      case 'b':
645
0
      case 'B':
646
0
        switch(state){
647
0
          case F_PARAM:
648
0
            state=BRANCH1;
649
0
            param->name.s=tmp;
650
0
            break;
651
0
          case GEN_PARAM:
652
0
            break;
653
0
          case F_CR:
654
0
          case F_LF:
655
0
          case F_CRLF:
656
0
            state=END_OF_HEADER;
657
0
            goto end_via;
658
0
          default:
659
0
            state=GEN_PARAM;
660
0
        }
661
0
        break;
662
0
      case 'p':
663
0
      case 'P':
664
0
        switch(state){
665
0
          case F_PARAM:
666
0
            state=GEN_PARAM;
667
0
            param->name.s=tmp;
668
0
            break;
669
0
          case RECEIVED1:
670
0
            state=RPORT1;
671
0
            break;
672
0
          case F_CR:
673
0
          case F_LF:
674
0
          case F_CRLF:
675
0
            state=END_OF_HEADER;
676
0
            goto end_via;
677
0
          default:
678
0
            state=GEN_PARAM;
679
0
        }
680
0
        break;
681
0
      case 'o':
682
0
      case 'O':
683
0
        switch(state){
684
0
          case F_PARAM:
685
0
            state=GEN_PARAM;
686
0
            param->name.s=tmp;
687
0
            break;
688
0
          case RPORT1:
689
0
            state=RPORT2;
690
0
            break;
691
0
          case F_CR:
692
0
          case F_LF:
693
0
          case F_CRLF:
694
0
            state=END_OF_HEADER;
695
0
            goto end_via;
696
0
          default:
697
0
            state=GEN_PARAM;
698
0
        }
699
0
        break;
700
0
      case 's':
701
0
      case 'S':
702
0
        switch(state){
703
0
          case F_PARAM:
704
0
            state=GEN_PARAM;
705
0
            param->name.s=tmp;
706
0
            break;
707
0
          case ALIAS4:
708
0
            state=FIN_ALIAS;
709
0
            break;
710
0
          case F_CR:
711
0
          case F_LF:
712
0
          case F_CRLF:
713
0
            state=END_OF_HEADER;
714
0
            goto end_via;
715
0
          default:
716
0
            state=GEN_PARAM;
717
0
        }
718
0
        break;
719
0
      default:
720
0
        switch(state){
721
0
          case F_PARAM:
722
0
            state=GEN_PARAM;
723
0
            param->name.s=tmp;
724
0
            break;
725
0
          case GEN_PARAM:
726
0
            break;
727
0
          case F_CR:
728
0
          case F_LF:
729
0
          case F_CRLF:
730
0
            state=END_OF_HEADER;
731
0
            goto end_via;
732
0
          default:
733
0
            state=GEN_PARAM;
734
0
        }
735
0
    }
736
0
  }/* for tmp*/
737
738
  /* end of packet? => error, no cr/lf,',' found!!!*/
739
0
  saved_state=state;
740
0
  state=END_OF_HEADER;
741
0
  goto parse_error;
742
743
0
 find_value:
744
0
  tmp++;
745
0
  for(;tmp<end;tmp++){
746
0
    switch(*tmp){
747
0
      case ' ':
748
0
      case '\t':
749
0
        switch(state){
750
0
          case L_VALUE:
751
0
          case F_VALUE: /*eat space*/
752
0
            break;
753
0
          case P_VALUE:
754
0
            state=L_PARAM;
755
0
            param->value.len=tmp-param->value.s;
756
0
            goto endofvalue;
757
0
          case P_STRING:
758
0
            break;
759
0
          case F_CR:
760
0
          case F_LF:
761
0
          case F_CRLF:
762
0
            state=saved_state;
763
0
            break;
764
0
          default:
765
0
            LM_ERR("invalid char <%c> in state %d\n", *tmp, state);
766
0
            goto parse_error;
767
0
        }
768
0
        break;
769
0
      case '\n':
770
0
        switch(state){
771
0
          case L_VALUE:
772
0
          case F_VALUE: /*eat space*/
773
0
          case P_STRING:
774
0
            saved_state=state;
775
0
            param->size=tmp-param->start;
776
0
            state=F_LF;
777
0
            break;
778
0
          case P_VALUE:
779
0
            saved_state=L_PARAM;
780
0
            state=F_LF;
781
0
            param->value.len=tmp-param->value.s;
782
0
            goto endofvalue;
783
0
          case F_LF:
784
0
          case F_CRLF:
785
0
            state=END_OF_HEADER;
786
0
            goto end_via;
787
0
          case F_CR:
788
0
            state=F_CRLF;
789
0
            break;
790
0
          default:
791
0
            LM_ERR("invalid char <%c> in state %d\n", *tmp, state);
792
0
            goto parse_error;
793
0
        }
794
0
        break;
795
0
      case '\r':
796
0
        switch(state){
797
0
          case L_VALUE:
798
0
          case F_VALUE: /*eat space*/
799
0
          case P_STRING:
800
0
            saved_state=state;
801
0
            param->size=tmp-param->start;
802
0
            state=F_CR;
803
0
            break;
804
0
          case P_VALUE:
805
0
            param->value.len=tmp-param->value.s;
806
0
            saved_state=L_PARAM;
807
0
            state=F_CR;
808
0
            goto endofvalue;
809
0
          case F_LF:
810
0
          case F_CR:
811
0
          case F_CRLF:
812
0
            state=END_OF_HEADER;
813
0
            goto end_via;
814
0
          default:
815
0
            LM_ERR("invalid char <%c> in state %d\n", *tmp, state);
816
0
            goto parse_error;
817
0
        }
818
0
        break;
819
820
0
      case '=':
821
0
        switch(state){
822
0
          case L_VALUE:
823
0
            state=F_VALUE;
824
0
            break;
825
0
          case P_STRING:
826
0
            break;
827
0
          case F_LF:
828
0
          case F_CR:
829
0
          case F_CRLF:
830
0
            state=END_OF_HEADER;
831
0
            goto end_via;
832
0
          default:
833
0
            LM_ERR("invalid char <%c> in state %d\n", *tmp, state);
834
0
            goto parse_error;
835
0
        }
836
0
        break;
837
0
      case ';':
838
0
        switch(state){
839
0
          case P_VALUE:
840
0
            param->value.len=tmp-param->value.s;
841
0
            state=F_PARAM;
842
0
            goto endofvalue;
843
0
          case F_VALUE:
844
0
            param->value.len=0;
845
0
            state=F_PARAM;
846
0
            goto endofvalue;
847
0
          case P_STRING:
848
0
            break; /* what to do? */
849
0
          case F_LF:
850
0
          case F_CR:
851
0
          case F_CRLF:
852
0
            state=END_OF_HEADER;
853
0
            goto end_via;
854
0
          case L_VALUE:
855
0
            if (param->type==FIN_RPORT){
856
0
              param->value.len=0;
857
0
              param->value.s=0; /* null value */
858
0
              state=F_PARAM;
859
0
              goto endofvalue;
860
0
            };
861
            /* no break */
862
0
          default:
863
0
            LM_ERR("invalid char <%c> in state %d\n", *tmp, state);
864
0
            goto parse_error;
865
0
        }
866
0
        break;
867
0
      case ',':
868
0
        switch(state){
869
0
          case P_VALUE:
870
0
            param->value.len=tmp-param->value.s;
871
0
            state=F_VIA;
872
0
            goto endofvalue;
873
0
          case P_STRING:
874
0
          case F_LF:
875
0
          case F_CR:
876
0
          case F_CRLF:
877
0
            state=END_OF_HEADER;
878
0
            goto end_via;
879
0
          case L_VALUE:
880
0
            if (param->type==FIN_RPORT){
881
0
              param->value.len=0;
882
0
              param->value.s=0; /* null value */
883
0
              state=F_VIA;
884
0
              goto endofvalue;
885
0
            };
886
            /* no break */
887
0
          default:
888
0
            LM_ERR("invalid char <%c> in state %d\n", *tmp, state);
889
0
            goto parse_error;
890
0
        }
891
0
        break; /* what to do? */
892
0
      case '"':
893
0
        switch(state){
894
0
          case F_VALUE:
895
0
            state=P_STRING;
896
0
            param->value.s=tmp+1;
897
0
            break;
898
0
          case P_STRING:
899
0
            state=L_PARAM;
900
0
            param->value.len=tmp-param->value.s;
901
0
            goto endofvalue;
902
0
          case F_LF:
903
0
          case F_CR:
904
0
          case F_CRLF:
905
0
            state=END_OF_HEADER;
906
0
            goto end_via;
907
0
          default:
908
0
            LM_ERR("invalid char <%c> in state %d\n", *tmp, state);
909
0
            goto parse_error;
910
0
        }
911
0
        break;
912
0
      case '\0':
913
0
        break;
914
915
0
      default:
916
0
        switch(state){
917
0
          case F_VALUE:
918
0
            state=P_VALUE;
919
0
            param->value.s=tmp;
920
0
            break;
921
0
          case P_VALUE:
922
0
          case P_STRING:
923
0
            break;
924
0
          case F_LF:
925
0
          case F_CR:
926
0
          case F_CRLF:
927
0
            state=END_OF_HEADER;
928
0
            goto end_via;
929
0
          default:
930
0
            LM_ERR("invalid char <%c> in state %d\n", *tmp, state);
931
0
            goto parse_error;
932
0
        }
933
0
    }
934
0
  } /* for2 tmp*/
935
936
  /* end of buff and no CR/LF =>error*/
937
0
  saved_state=state;
938
0
  state=END_OF_HEADER;
939
0
  goto parse_error;
940
941
0
 endofparam:
942
0
 endofvalue:
943
0
  param->size=tmp-param->start;
944
0
normal_exit:
945
0
  *pstate=state;
946
0
  *psaved_state=saved_state;
947
0
  LM_DBG("found param type %d, <%.*s> = <%.*s>; state=%d\n", param->type,
948
0
      param->name.len, ZSW(param->name.s),
949
0
      (param->value.len?param->value.len:3),
950
0
      (param->value.len?param->value.s:"n/a"), state);
951
0
  return tmp;
952
953
0
 end_via:
954
       /* if we are here we found an "unexpected" end of via
955
        *  (cr/lf). This is valid only if the param type is GEN_PARAM or
956
      *  RPORT (the only ones which can miss the value; HIDDEN is a
957
      *  special case )*/
958
0
  if ((param->type==GEN_PARAM)||(param->type==PARAM_RPORT)){
959
0
    saved_state=L_PARAM; /* change the saved_state, we have an unknown
960
                            param. w/o a value */
961
    /* param->size should be computed before */
962
0
    goto normal_exit;
963
0
  }
964
0
  *pstate=state;
965
0
  *psaved_state=saved_state;
966
0
  LM_DBG("error on  param type %d, <%.*s>, state=%d, saved_state=%d\n",
967
0
    param->type, param->name.len, ZSW(param->name.s), state, saved_state);
968
969
0
parse_error:
970
0
  LM_ERR("parse_via_param\n");
971
0
  param->type=PARAM_ERROR;
972
0
  *pstate=PARAM_ERROR;
973
0
  *psaved_state=state;
974
0
  return tmp;
975
0
}
976
977
978
979
/*
980
 * call it with a vb initialized to 0
981
 * returns: pointer after the parsed parts and sets vb->error
982
 * WARNING: don't forget to cleanup on error with free_via_list(vb)!
983
 */
984
char* parse_via(char* buffer, char* end, struct via_body *vbody)
985
0
{
986
0
  char* tmp;
987
0
  char* param_start;
988
0
  unsigned char state;
989
0
  unsigned char saved_state;
990
0
  int c_nest;
991
0
  int err;
992
0
  struct via_body* vb;
993
0
  struct via_param* param;
994
995
0
  vb=vbody; /* keep orignal vbody value, needed to set the error member
996
         in case of multiple via bodies in the same header */
997
0
parse_again:
998
0
  vb->error=PARSE_ERROR;
999
  /* parse start of via ( SIP/2.0/UDP    )*/
1000
0
  state=F_SIP;
1001
0
  saved_state=0; /*it should generate error if it's used without set*/
1002
0
  param_start=0;
1003
0
  for(tmp=buffer;tmp<end;tmp++){
1004
0
    switch(*tmp){
1005
0
      case ' ':
1006
0
      case'\t':
1007
0
        switch(state){
1008
0
          case L_VER: /* eat space */
1009
0
          case L_PROTO:
1010
0
          case F_SIP:
1011
0
          case F_VER:
1012
0
          case F_PROTO:
1013
0
            break;
1014
0
          case FIN_UDP:
1015
0
            vb->transport.len=tmp-vb->transport.s;
1016
0
            vb->proto=PROTO_UDP;
1017
0
            state=F_HOST; /* start looking for host*/
1018
0
            goto main_via;
1019
0
          case FIN_TCP:
1020
            /* finished proto parsing */
1021
0
            vb->transport.len=tmp-vb->transport.s;
1022
0
            vb->proto=PROTO_TCP;
1023
0
            state=F_HOST; /* start looking for host*/
1024
0
            goto main_via;
1025
0
          case FIN_TLS:
1026
            /* finished proto parsing */
1027
0
            vb->transport.len=tmp-vb->transport.s;
1028
0
            vb->proto=PROTO_TLS;
1029
0
            state=F_HOST; /* start looking for host*/
1030
0
            goto main_via;
1031
0
          case FIN_SCTP:
1032
            /* finished proto parsing */
1033
0
            vb->transport.len=tmp-vb->transport.s;
1034
0
            vb->proto=PROTO_SCTP;
1035
0
            state=F_HOST; /* start looking for host*/
1036
0
            goto main_via;
1037
0
          case FIN_WS:
1038
0
          case WS_WSS:
1039
            /* finished proto parsing */
1040
0
            vb->transport.len=tmp-vb->transport.s;
1041
0
            vb->proto=PROTO_WS;
1042
0
            state=F_HOST; /* start looking for host*/
1043
0
            goto main_via;
1044
0
          case FIN_WSS:
1045
            /* finished proto parsing */
1046
0
            vb->transport.len=tmp-vb->transport.s;
1047
0
            vb->proto=PROTO_WSS;
1048
0
            state=F_HOST; /* start looking for host*/
1049
0
            goto main_via;
1050
0
          case OTHER_PROTO:
1051
            /* finished proto parsing */
1052
0
            vb->transport.len=tmp-vb->transport.s;
1053
0
            vb->proto=PROTO_OTHER;
1054
0
            state=F_HOST; /* start looking for host*/
1055
0
            goto main_via;
1056
0
          case FIN_SIP:
1057
0
            vb->name.len=tmp-vb->name.s;
1058
0
            state=L_VER;
1059
0
            break;
1060
0
          case FIN_VER:
1061
0
            vb->version.len=tmp-vb->version.s;
1062
0
            state=L_PROTO;
1063
0
            break;
1064
0
          case F_LF:
1065
0
          case F_CRLF:
1066
0
          case F_CR: /* header continues on this line */
1067
0
            state=saved_state;
1068
0
            break;
1069
0
          default:
1070
0
            LM_ERR("bad char <%c> on state %d\n", *tmp, state);
1071
0
            goto parse_error;
1072
0
        }
1073
0
        break;
1074
0
      case '\n':
1075
0
        switch(state){
1076
0
          case L_VER:
1077
0
          case F_SIP:
1078
0
          case F_VER:
1079
0
          case F_PROTO:
1080
0
          case L_PROTO:
1081
0
            saved_state=state;
1082
0
            state=F_LF;
1083
0
            break;
1084
0
          case FIN_UDP:
1085
0
            vb->transport.len=tmp-vb->transport.s;
1086
0
            vb->proto=PROTO_UDP;
1087
0
            state=F_LF;
1088
0
            saved_state=F_HOST; /* start looking for host*/
1089
0
            goto main_via;
1090
0
          case FIN_TCP:
1091
0
            vb->transport.len=tmp-vb->transport.s;
1092
0
            vb->proto=PROTO_TCP;
1093
0
            state=F_LF;
1094
0
            saved_state=F_HOST; /* start looking for host*/
1095
0
            goto main_via;
1096
0
          case FIN_TLS:
1097
0
            vb->transport.len=tmp-vb->transport.s;
1098
0
            vb->proto=PROTO_TLS;
1099
0
            state=F_LF;
1100
0
            saved_state=F_HOST; /* start looking for host*/
1101
0
            goto main_via;
1102
0
          case WS_WSS:
1103
0
          case FIN_WS:
1104
0
            vb->transport.len=tmp-vb->transport.s;
1105
0
            vb->proto=PROTO_WS;
1106
0
            state=F_LF;
1107
0
            saved_state=F_HOST; /* start looking for host*/
1108
0
            goto main_via;
1109
0
          case FIN_WSS:
1110
0
            vb->transport.len=tmp-vb->transport.s;
1111
0
            vb->proto=PROTO_WS;
1112
0
            state=F_LF;
1113
0
            saved_state=F_HOST; /* start looking for host*/
1114
0
            goto main_via;
1115
0
          case OTHER_PROTO:
1116
            /* finished proto parsing */
1117
0
            vb->transport.len=tmp-vb->transport.s;
1118
0
            vb->proto=PROTO_OTHER;
1119
0
            state=F_LF;
1120
0
            saved_state=F_HOST; /* start looking for host*/
1121
0
            goto main_via;
1122
0
          case FIN_SIP:
1123
0
            vb->name.len=tmp-vb->name.s;
1124
0
            state=F_LF;
1125
0
            saved_state=L_VER;
1126
0
            break;
1127
0
          case FIN_VER:
1128
0
            vb->version.len=tmp-vb->version.s;
1129
0
            state=F_LF;
1130
0
            saved_state=L_PROTO;
1131
0
            break;
1132
0
          case F_CR:
1133
0
            state=F_CRLF;
1134
0
            break;
1135
0
          case F_LF:
1136
0
          case F_CRLF:
1137
0
            state=saved_state;
1138
0
            goto endofheader;
1139
0
          default:
1140
0
            LM_ERR("bad char <%c> on state %d\n", *tmp, state);
1141
0
            goto parse_error;
1142
0
        }
1143
0
        break;
1144
0
      case '\r':
1145
0
        switch(state){
1146
0
          case L_VER:
1147
0
          case F_SIP:
1148
0
          case F_VER:
1149
0
          case F_PROTO:
1150
0
          case L_PROTO:
1151
0
            saved_state=state;
1152
0
            state=F_CR;
1153
0
            break;
1154
0
          case FIN_UDP:
1155
0
            vb->transport.len=tmp-vb->transport.s;
1156
0
            vb->proto=PROTO_UDP;
1157
0
            state=F_CR;
1158
0
            saved_state=F_HOST;
1159
0
            goto main_via;
1160
0
          case FIN_TCP:
1161
0
            vb->transport.len=tmp-vb->transport.s;
1162
0
            vb->proto=PROTO_TCP;
1163
0
            state=F_CR;
1164
0
            saved_state=F_HOST;
1165
0
            goto main_via;
1166
0
          case FIN_TLS:
1167
0
            vb->transport.len=tmp-vb->transport.s;
1168
0
            vb->proto=PROTO_TLS;
1169
0
            state=F_CR;
1170
0
            saved_state=F_HOST;
1171
0
            goto main_via;
1172
0
          case WS_WSS:
1173
0
          case FIN_WS:
1174
0
            vb->transport.len=tmp-vb->transport.s;
1175
0
            vb->proto=PROTO_WS;
1176
0
            state=F_CR;
1177
0
            saved_state=F_HOST;
1178
0
            goto main_via;
1179
0
          case FIN_WSS:
1180
0
            vb->transport.len=tmp-vb->transport.s;
1181
0
            vb->proto=PROTO_WSS;
1182
0
            state=F_CR;
1183
0
            saved_state=F_HOST;
1184
0
            goto main_via;
1185
0
          case OTHER_PROTO:
1186
0
            vb->transport.len=tmp-vb->transport.s;
1187
0
            vb->proto=PROTO_OTHER;
1188
0
            state=F_CR;
1189
0
            saved_state=F_HOST;
1190
0
            goto main_via;
1191
0
          case FIN_SIP:
1192
0
            vb->name.len=tmp-vb->name.s;
1193
0
            state=F_CR;
1194
0
            saved_state=L_VER;
1195
0
            break;
1196
0
          case FIN_VER:
1197
0
            vb->version.len=tmp-vb->version.s;
1198
0
            state=F_CR;
1199
0
            saved_state=L_PROTO;
1200
0
            break;
1201
0
          case F_LF: /*end of line ?next header?*/
1202
0
          case F_CR:
1203
0
          case F_CRLF:
1204
0
            state=saved_state;
1205
0
            goto endofheader;
1206
0
          default:
1207
0
            LM_ERR("bad char <%c> on state %d\n", *tmp, state);
1208
0
            goto parse_error;
1209
0
        }
1210
0
        break;
1211
1212
0
      case '/':
1213
0
        switch(state){
1214
0
          case FIN_SIP:
1215
0
            vb->name.len=tmp-vb->name.s;
1216
0
            state=F_VER;
1217
0
            break;
1218
0
          case FIN_VER:
1219
0
            vb->version.len=tmp-vb->version.s;
1220
0
            state=F_PROTO;
1221
0
            break;
1222
0
          case L_VER:
1223
0
            state=F_VER;
1224
0
            break;
1225
0
          case L_PROTO:
1226
0
            state=F_PROTO;
1227
0
            break;
1228
0
          default:
1229
0
            LM_ERR("bad char <%c> on state %d\n", *tmp, state);
1230
0
            goto parse_error;
1231
0
        }
1232
0
        break;
1233
        /* match SIP*/
1234
0
      case 'S':
1235
0
      case 's':
1236
0
        switch(state){
1237
0
          case F_SIP:
1238
0
            state=SIP1;
1239
0
            vb->name.s=tmp;
1240
0
            break;
1241
0
          case TLS2:
1242
0
            state=FIN_TLS;
1243
0
            break;
1244
0
          case F_PROTO:
1245
0
            state=SCTP1;
1246
0
            vb->transport.s=tmp;
1247
0
            break;
1248
0
          case WS1:
1249
0
            state=WS_WSS;
1250
0
            break;
1251
0
          case WS_WSS:
1252
0
            state=FIN_WSS;
1253
0
            break;
1254
0
          case OTHER_PROTO:
1255
0
            break;
1256
0
          case UDP1:
1257
0
          case UDP2:
1258
0
          case FIN_UDP:
1259
0
          case TCP_TLS1:
1260
0
          case TCP2:
1261
0
          case FIN_TCP:
1262
0
          case FIN_TLS:
1263
0
          case SCTP1:
1264
0
          case SCTP2:
1265
0
          case SCTP3:
1266
0
          case FIN_SCTP:
1267
0
          case FIN_WS:
1268
0
          case FIN_WSS:
1269
0
            state=OTHER_PROTO;
1270
0
            break;
1271
0
          default:
1272
0
            LM_ERR("bad char <%c> on state %d\n", *tmp, state);
1273
0
            goto parse_error;
1274
0
        }
1275
0
        break;
1276
0
      case 'I':
1277
0
      case 'i':
1278
0
        switch(state){
1279
0
          case SIP1:
1280
0
            state=SIP2;
1281
0
            break;
1282
0
          case OTHER_PROTO:
1283
0
            break;
1284
0
          case UDP1:
1285
0
          case UDP2:
1286
0
          case FIN_UDP:
1287
0
          case TCP_TLS1:
1288
0
          case TCP2:
1289
0
          case FIN_TCP:
1290
0
          case TLS2:
1291
0
          case FIN_TLS:
1292
0
          case SCTP1:
1293
0
          case SCTP2:
1294
0
          case SCTP3:
1295
0
          case FIN_SCTP:
1296
0
          case WS1:
1297
0
          case WS_WSS:
1298
0
          case FIN_WS:
1299
0
          case FIN_WSS:
1300
0
            state=OTHER_PROTO;
1301
0
            break;
1302
0
          default:
1303
0
            LM_ERR("bad char <%c> on state %d\n", *tmp, state);
1304
0
            goto parse_error;
1305
0
        }
1306
0
        break;
1307
0
      case 'p':
1308
0
      case 'P':
1309
0
        switch(state){
1310
0
          case SIP2:
1311
0
            state=FIN_SIP;
1312
0
            break;
1313
          /* allow p in PROTO */
1314
0
          case UDP2:
1315
0
            state=FIN_UDP;
1316
0
            break;
1317
0
          case TCP2:
1318
0
            state=FIN_TCP;
1319
0
            break;
1320
0
          case SCTP3:
1321
0
            state=FIN_SCTP;
1322
0
            break;
1323
0
          case OTHER_PROTO:
1324
0
            break;
1325
0
          case UDP1:
1326
0
          case FIN_UDP:
1327
0
          case TCP_TLS1:
1328
0
          case FIN_TCP:
1329
0
          case TLS2:
1330
0
          case FIN_TLS:
1331
0
          case SCTP1:
1332
0
          case SCTP2:
1333
0
          case FIN_SCTP:
1334
0
          case WS1:
1335
0
          case WS_WSS:
1336
0
          case FIN_WS:
1337
0
          case FIN_WSS:
1338
0
            state=OTHER_PROTO;
1339
0
            break;
1340
0
          default:
1341
0
            LM_ERR("bad char <%c> on state %d\n", *tmp, state);
1342
0
            goto parse_error;
1343
0
        }
1344
0
        break;
1345
0
      case 'U':
1346
0
      case 'u':
1347
0
        switch(state){
1348
0
          case F_PROTO:
1349
0
            state=UDP1;
1350
0
            vb->transport.s=tmp;
1351
0
            break;
1352
0
          case OTHER_PROTO:
1353
0
            break;
1354
0
          case UDP1:
1355
0
          case UDP2:
1356
0
          case FIN_UDP:
1357
0
          case TCP_TLS1:
1358
0
          case TCP2:
1359
0
          case FIN_TCP:
1360
0
          case TLS2:
1361
0
          case FIN_TLS:
1362
0
          case SCTP1:
1363
0
          case SCTP2:
1364
0
          case SCTP3:
1365
0
          case FIN_SCTP:
1366
0
          case WS1:
1367
0
          case WS_WSS:
1368
0
          case FIN_WS:
1369
0
          case FIN_WSS:
1370
0
            state=OTHER_PROTO;
1371
0
            break;
1372
0
          default:
1373
0
            LM_ERR("bad char <%c> on state %d\n", *tmp, state);
1374
0
            goto parse_error;
1375
0
        }
1376
0
        break;
1377
0
      case 'D':
1378
0
      case 'd':
1379
0
        switch(state){
1380
0
          case UDP1:
1381
0
            state=UDP2;
1382
0
            break;
1383
0
          case OTHER_PROTO:
1384
0
            break;
1385
0
          case UDP2:
1386
0
          case FIN_UDP:
1387
0
          case TCP_TLS1:
1388
0
          case TCP2:
1389
0
          case FIN_TCP:
1390
0
          case TLS2:
1391
0
          case FIN_TLS:
1392
0
          case SCTP1:
1393
0
          case SCTP2:
1394
0
          case SCTP3:
1395
0
          case FIN_SCTP:
1396
0
          case WS1:
1397
0
          case WS_WSS:
1398
0
          case FIN_WS:
1399
0
          case FIN_WSS:
1400
0
            state=OTHER_PROTO;
1401
0
            break;
1402
0
          default:
1403
0
            LM_ERR("bad char <%c> on state %d\n", *tmp, state);
1404
0
            goto parse_error;
1405
0
        }
1406
0
        break;
1407
0
      case 'T':
1408
0
      case 't':
1409
0
        switch(state){
1410
0
          case F_PROTO:
1411
0
            state=TCP_TLS1;
1412
0
            vb->transport.s=tmp;
1413
0
            break;
1414
0
          case SCTP2:
1415
0
            state=SCTP3;
1416
0
            break;
1417
0
          case OTHER_PROTO:
1418
0
            break;
1419
0
          case UDP1:
1420
0
          case UDP2:
1421
0
          case FIN_UDP:
1422
0
          case TCP_TLS1:
1423
0
          case TCP2:
1424
0
          case FIN_TCP:
1425
0
          case TLS2:
1426
0
          case FIN_TLS:
1427
0
          case SCTP1:
1428
0
          case SCTP3:
1429
0
          case FIN_SCTP:
1430
0
          case WS1:
1431
0
          case WS_WSS:
1432
0
          case FIN_WS:
1433
0
          case FIN_WSS:
1434
0
            state=OTHER_PROTO;
1435
0
            break;
1436
0
          default:
1437
0
            LM_ERR("bad char <%c> on state %d\n", *tmp, state);
1438
0
            goto parse_error;
1439
0
        }
1440
0
        break;
1441
0
      case 'C':
1442
0
      case 'c':
1443
0
        switch(state){
1444
0
          case TCP_TLS1:
1445
0
            state=TCP2;
1446
0
            break;
1447
0
          case SCTP1:
1448
0
            state=SCTP2;
1449
0
            break;
1450
0
          case OTHER_PROTO:
1451
0
            break;
1452
0
          case UDP1:
1453
0
          case UDP2:
1454
0
          case FIN_UDP:
1455
0
          case TCP2:
1456
0
          case FIN_TCP:
1457
0
          case TLS2:
1458
0
          case FIN_TLS:
1459
0
          case SCTP2:
1460
0
          case SCTP3:
1461
0
          case FIN_SCTP:
1462
0
          case WS1:
1463
0
          case WS_WSS:
1464
0
          case FIN_WS:
1465
0
          case FIN_WSS:
1466
0
            state=OTHER_PROTO;
1467
0
            break;
1468
0
          default:
1469
0
            LM_ERR("bad char <%c> on state %d\n", *tmp, state);
1470
0
            goto parse_error;
1471
0
        }
1472
0
        break;
1473
0
      case 'L':
1474
0
      case 'l':
1475
0
        switch(state){
1476
0
          case TCP_TLS1:
1477
0
            state=TLS2;
1478
0
            break;
1479
0
          case OTHER_PROTO:
1480
0
            break;
1481
0
          case UDP1:
1482
0
          case UDP2:
1483
0
          case FIN_UDP:
1484
0
          case TCP2:
1485
0
          case FIN_TCP:
1486
0
          case TLS2:
1487
0
          case FIN_TLS:
1488
0
          case SCTP1:
1489
0
          case SCTP2:
1490
0
          case SCTP3:
1491
0
          case FIN_SCTP:
1492
0
          case WS1:
1493
0
          case WS_WSS:
1494
0
          case FIN_WS:
1495
0
          case FIN_WSS:
1496
0
            state=OTHER_PROTO;
1497
0
            break;
1498
0
          default:
1499
0
            LM_ERR("bad char <%c> on state %d\n", *tmp, state);
1500
0
            goto parse_error;
1501
0
        }
1502
0
        break;
1503
0
      case 'W':
1504
0
      case 'w':
1505
0
        switch(state){
1506
0
          case F_PROTO:
1507
0
            state=WS1;
1508
0
            vb->transport.s=tmp;
1509
0
            break;
1510
0
          case OTHER_PROTO:
1511
0
            break;
1512
0
          case UDP1:
1513
0
          case UDP2:
1514
0
          case FIN_UDP:
1515
0
          case TCP_TLS1:
1516
0
          case TCP2:
1517
0
          case FIN_TCP:
1518
0
          case TLS2:
1519
0
          case FIN_TLS:
1520
0
          case SCTP1:
1521
0
          case SCTP2:
1522
0
          case SCTP3:
1523
0
          case FIN_SCTP:
1524
0
          case WS1:
1525
0
          case WS_WSS:
1526
0
          case FIN_WS:
1527
0
          case FIN_WSS:
1528
0
            state=OTHER_PROTO;
1529
0
            break;
1530
0
          default:
1531
0
            LM_ERR("bad char <%c> on state %d\n", *tmp, state);
1532
0
            goto parse_error;
1533
0
        }
1534
0
        break;
1535
      /*match 2.0*/
1536
0
      case '2':
1537
0
        switch(state){
1538
0
          case F_VER:
1539
0
            state=VER1;
1540
0
            vb->version.s=tmp;
1541
0
            break;
1542
0
          case OTHER_PROTO:
1543
0
            break;
1544
0
          case UDP1:
1545
0
          case UDP2:
1546
0
          case FIN_UDP:
1547
0
          case TCP_TLS1:
1548
0
          case TCP2:
1549
0
          case FIN_TCP:
1550
0
          case TLS2:
1551
0
          case FIN_TLS:
1552
0
          case SCTP1:
1553
0
          case SCTP2:
1554
0
          case SCTP3:
1555
0
          case FIN_SCTP:
1556
0
          case WS1:
1557
0
          case WS_WSS:
1558
0
          case FIN_WS:
1559
0
          case FIN_WSS:
1560
0
            state=OTHER_PROTO;
1561
0
            break;
1562
0
          default:
1563
0
            LM_ERR("bad char <%c> on state %d\n", *tmp, state);
1564
0
            goto parse_error;
1565
0
        }
1566
0
        break;
1567
0
      case '.':
1568
0
        switch(state){
1569
0
          case VER1:
1570
0
            state=VER2;
1571
0
            break;
1572
0
          case OTHER_PROTO:
1573
0
            break;
1574
0
          case UDP1:
1575
0
          case UDP2:
1576
0
          case FIN_UDP:
1577
0
          case TCP_TLS1:
1578
0
          case TCP2:
1579
0
          case FIN_TCP:
1580
0
          case TLS2:
1581
0
          case FIN_TLS:
1582
0
          case SCTP1:
1583
0
          case SCTP2:
1584
0
          case SCTP3:
1585
0
          case FIN_SCTP:
1586
0
          case WS1:
1587
0
          case WS_WSS:
1588
0
          case FIN_WS:
1589
0
          case FIN_WSS:
1590
0
            state=OTHER_PROTO;
1591
0
            break;
1592
0
          default:
1593
0
            LM_ERR("bad char <%c> on state %d\n", *tmp, state);
1594
0
            goto parse_error;
1595
0
        }
1596
0
         break;
1597
0
      case '0':
1598
0
        switch(state){
1599
0
          case VER2:
1600
0
            state=FIN_VER;
1601
0
            break;
1602
0
          case OTHER_PROTO:
1603
0
            break;
1604
0
          case UDP1:
1605
0
          case UDP2:
1606
0
          case FIN_UDP:
1607
0
          case TCP_TLS1:
1608
0
          case TCP2:
1609
0
          case FIN_TCP:
1610
0
          case TLS2:
1611
0
          case FIN_TLS:
1612
0
          case SCTP1:
1613
0
          case SCTP2:
1614
0
          case SCTP3:
1615
0
          case FIN_SCTP:
1616
0
          case WS1:
1617
0
          case WS_WSS:
1618
0
          case FIN_WS:
1619
0
          case FIN_WSS:
1620
0
            state=OTHER_PROTO;
1621
0
            break;
1622
0
          default:
1623
0
            LM_ERR("bad char <%c> on state %d\n", *tmp, state);
1624
0
            goto parse_error;
1625
0
        }
1626
0
        break;
1627
1628
0
      default:
1629
0
        switch(state){
1630
0
          case F_PROTO:
1631
0
            state=OTHER_PROTO;
1632
0
            vb->transport.s=tmp;
1633
0
            break;
1634
0
          case OTHER_PROTO:
1635
0
            break;
1636
0
          case UDP1:
1637
0
          case UDP2:
1638
0
          case FIN_UDP:
1639
0
          case TCP_TLS1:
1640
0
          case TCP2:
1641
0
          case FIN_TCP:
1642
0
          case TLS2:
1643
0
          case FIN_TLS:
1644
0
          case SCTP1:
1645
0
          case SCTP2:
1646
0
          case SCTP3:
1647
0
          case FIN_SCTP:
1648
0
          case WS1:
1649
0
          case WS_WSS:
1650
0
          case FIN_WS:
1651
0
          case FIN_WSS:
1652
0
            state=OTHER_PROTO;
1653
0
            break;
1654
0
          default:
1655
0
            LM_ERR("bad char <%c> on state %d\n", *tmp, state);
1656
0
            goto parse_error;
1657
0
        }
1658
0
        break;
1659
0
    }
1660
0
  } /* for tmp*/
1661
1662
  /* we should not be here! if everything is ok > main_via*/
1663
0
  LM_ERR("bad via: end of packet on state=%d\n", state);
1664
0
  goto parse_error;
1665
1666
0
 main_via:
1667
  /* inc tmp to point to the next char*/
1668
0
  tmp++;
1669
0
  c_nest=0;
1670
0
  /*state should always be F_HOST here*/;
1671
0
  for(;tmp<end;tmp++){
1672
0
    switch(*tmp){
1673
0
    case ' ':
1674
0
    case '\t':
1675
0
      switch(state){
1676
0
          case F_HOST:/*eat the spaces*/
1677
0
            break;
1678
0
          case P_HOST:
1679
             /*mark end of host*/
1680
0
             vb->host.len=tmp-vb->host.s;
1681
0
             state=L_PORT;
1682
0
             break;
1683
0
          case L_PORT: /*eat the spaces*/
1684
0
          case F_PORT:
1685
0
            break;
1686
0
          case P_PORT:
1687
            /*end of port */
1688
0
            vb->port_str.len=tmp-vb->port_str.s;
1689
0
            state=L_PARAM;
1690
0
            break;
1691
0
          case L_PARAM: /* eat the space */
1692
0
          case F_PARAM:
1693
0
            break;
1694
0
          case P_PARAM:
1695
          /*  *tmp=0;*/ /*!?end of param*/
1696
0
            state=L_PARAM;
1697
0
            break;
1698
0
          case L_VIA:
1699
0
          case F_VIA: /* eat the space */
1700
0
            break;
1701
0
          case F_COMMENT:
1702
0
          case P_COMMENT:
1703
0
            break;
1704
0
          case F_IP6HOST: /*no spaces allowed*/
1705
0
          case P_IP6HOST:
1706
0
            LM_ERR("bad ipv6 reference\n");
1707
0
            goto parse_error;
1708
0
          case F_CRLF:
1709
0
          case F_LF:
1710
0
          case F_CR:
1711
            /*previous=crlf and now =' '*/
1712
0
            state=saved_state;
1713
0
            break;
1714
0
          default:
1715
0
            LM_ERR("on <%c>, state=%d\n",*tmp, state);
1716
0
            goto parse_error;
1717
0
        }
1718
0
      break;
1719
0
      case '\n':
1720
0
        switch(state){
1721
0
          case F_HOST:/*eat the spaces*/
1722
0
          case L_PORT: /*eat the spaces*/
1723
0
          case F_PORT:
1724
0
          case L_PARAM: /* eat the space */
1725
0
          case F_PARAM:
1726
0
          case F_VIA: /* eat the space */
1727
0
          case L_VIA:
1728
0
          case F_COMMENT:
1729
0
          case P_COMMENT:
1730
0
          case F_IP6HOST:
1731
0
          case P_IP6HOST:
1732
0
            saved_state=state;
1733
0
            state=F_LF;
1734
0
            break;
1735
0
          case P_HOST:
1736
             /*mark end of host*/
1737
0
             vb->host.len=tmp-vb->host.s;
1738
0
             saved_state=L_PORT;
1739
0
             state=F_LF;
1740
0
             break;
1741
0
          case P_PORT:
1742
            /*end of port */
1743
0
            vb->port_str.len=tmp-vb->port_str.s;
1744
0
            saved_state=L_PARAM;
1745
0
            state=F_LF;
1746
0
            break;
1747
0
          case P_PARAM:
1748
          /*  *tmp=0;*/ /*!?end of param*/
1749
0
            saved_state=L_PARAM;
1750
0
            state=F_LF;
1751
0
            break;
1752
0
          case F_CR:
1753
0
            state=F_CRLF;
1754
0
            break;
1755
0
          case F_CRLF:
1756
0
          case F_LF:
1757
0
            state=saved_state;
1758
0
            goto endofheader;
1759
0
          default:
1760
0
            LM_CRIT("BUG on <%c>\n",*tmp);
1761
0
            goto  parse_error;
1762
0
        }
1763
0
      break;
1764
0
    case '\r':
1765
0
        switch(state){
1766
0
          case F_HOST:/*eat the spaces*/
1767
0
          case L_PORT: /*eat the spaces*/
1768
0
          case F_PORT:
1769
0
          case L_PARAM: /* eat the space */
1770
0
          case F_PARAM:
1771
0
          case F_VIA: /* eat the space */
1772
0
          case L_VIA:
1773
0
          case F_COMMENT:
1774
0
          case P_COMMENT:
1775
0
          case F_IP6HOST:
1776
0
          case P_IP6HOST:
1777
0
            saved_state=state;
1778
0
            state=F_CR;
1779
0
            break;
1780
0
          case P_HOST:
1781
             /*mark end of host*/
1782
0
             vb->host.len=tmp-vb->host.s;
1783
0
             saved_state=L_PORT;
1784
0
             state=F_CR;
1785
0
             break;
1786
0
          case P_PORT:
1787
            /*end of port */
1788
0
            vb->port_str.len=tmp-vb->port_str.s;
1789
0
            saved_state=L_PARAM;
1790
0
            state=F_CR;
1791
0
            break;
1792
0
          case P_PARAM:
1793
          /*  *tmp=0;*/ /*!?end of param*/
1794
0
            saved_state=L_PARAM;
1795
0
            state=F_CR;
1796
0
            break;
1797
0
          case F_CRLF:
1798
0
          case F_CR:
1799
0
          case F_LF:
1800
0
            state=saved_state;
1801
0
            goto endofheader;
1802
0
          default:
1803
0
            LM_ERR("on <%c>\n",*tmp);
1804
0
            goto parse_error;
1805
0
        }
1806
0
      break;
1807
1808
0
      case ':':
1809
0
        switch(state){
1810
0
          case F_IP6HOST:
1811
0
            state=P_IP6HOST;
1812
0
            break;
1813
0
          case P_IP6HOST:
1814
0
            break;
1815
0
          case P_HOST:
1816
            /*mark  end of host*/
1817
0
            vb->host.len=tmp-vb->host.s;
1818
0
            state=F_PORT;
1819
0
            break;
1820
0
          case L_PORT:
1821
0
            state=F_PORT;
1822
0
            break;
1823
0
          case P_PORT:
1824
0
            LM_ERR("bad port\n");
1825
0
            goto parse_error;
1826
0
          case L_PARAM:
1827
0
          case F_PARAM:
1828
0
          case P_PARAM:
1829
0
            LM_ERR("bad char <%c> in state %d\n",
1830
0
              *tmp,state);
1831
0
            goto parse_error;
1832
0
          case L_VIA:
1833
0
          case F_VIA:
1834
0
            LM_ERR("bad char in compact via\n");
1835
0
            goto parse_error;
1836
0
          case F_CRLF:
1837
0
          case F_LF:
1838
0
          case F_CR:
1839
            /*previous=crlf and now !=' '*/
1840
0
            goto endofheader;
1841
0
          case F_COMMENT:/*everything is allowed in a comment*/
1842
0
            vb->comment.s=tmp;
1843
0
            state=P_COMMENT;
1844
0
            break;
1845
0
          case P_COMMENT: /*everything is allowed in a comment*/
1846
0
            break;
1847
0
          default:
1848
0
            LM_ERR("on <%c> state %d\n", *tmp, state);
1849
0
            goto parse_error;
1850
0
        }
1851
0
        break;
1852
0
      case ';':
1853
0
        switch(state){
1854
0
          case F_HOST:
1855
0
          case F_IP6HOST:
1856
0
            LM_ERR(" no host found\n");
1857
0
            goto parse_error;
1858
0
          case P_IP6HOST:
1859
0
            LM_ERR(" bad ipv6 reference\n");
1860
0
            goto parse_error;
1861
0
          case P_HOST:
1862
0
            vb->host.len=tmp-vb->host.s;
1863
0
            state=F_PARAM;
1864
0
            param_start=tmp+1;
1865
0
            break;
1866
0
          case P_PORT:
1867
            /*mark the end*/
1868
0
            vb->port_str.len=tmp-vb->port_str.s;
1869
            /* fall through */
1870
0
          case L_PORT:
1871
0
          case L_PARAM:
1872
0
            state=F_PARAM;
1873
0
            param_start=tmp+1;
1874
0
            break;
1875
0
          case F_PORT:
1876
0
            LM_ERR(" bad char <%c> in state %d\n",
1877
0
              *tmp,state);
1878
0
            goto parse_error;
1879
0
          case F_PARAM:
1880
0
            LM_ERR("null param?\n");
1881
0
            goto parse_error;
1882
0
          case P_PARAM:
1883
            /*hmm next, param?*/
1884
0
            state=F_PARAM;
1885
0
            param_start=tmp+1;
1886
0
            break;
1887
0
          case L_VIA:
1888
0
          case F_VIA:
1889
0
            LM_ERR("bad char <%c> in next via\n", *tmp);
1890
0
            goto parse_error;
1891
0
          case F_CRLF:
1892
0
          case F_LF:
1893
0
          case F_CR:
1894
            /*previous=crlf and now !=' '*/
1895
0
            goto endofheader;
1896
0
          case F_COMMENT:/*everything is allowed in a comment*/
1897
0
            vb->comment.s=tmp;
1898
0
            state=P_COMMENT;
1899
0
            break;
1900
0
          case P_COMMENT: /*everything is allowed in a comment*/
1901
0
            break;
1902
1903
0
          default:
1904
0
            LM_ERR("on <%c> state %d\n", *tmp, state);
1905
0
            goto parse_error;
1906
0
        }
1907
0
      break;
1908
0
      case ',':
1909
0
        switch(state){
1910
0
          case F_HOST:
1911
0
          case F_IP6HOST:
1912
0
            LM_ERR("no host found\n");
1913
0
            goto parse_error;
1914
0
          case P_IP6HOST:
1915
0
            LM_ERR(" bad ipv6 reference\n");
1916
0
            goto parse_error;
1917
0
          case P_HOST:
1918
            /*mark the end*/
1919
0
            vb->host.len=tmp-vb->host.s;
1920
0
            state=F_VIA;
1921
0
            break;
1922
0
          case P_PORT:
1923
            /*mark the end*/
1924
0
            vb->port_str.len=tmp-vb->port_str.s;
1925
0
            state=F_VIA;
1926
0
            break;
1927
0
          case L_PORT:
1928
0
          case L_PARAM:
1929
0
          case P_PARAM:
1930
0
          case L_VIA:
1931
0
            state=F_VIA;
1932
0
            break;
1933
0
          case F_PORT:
1934
0
          case F_PARAM:
1935
0
            LM_ERR("invalid char <%c> in state %d\n", *tmp,state);
1936
0
            goto parse_error;
1937
0
          case F_VIA:
1938
            /* do  nothing,  eat ","*/
1939
0
            break;
1940
0
          case F_CRLF:
1941
0
          case F_LF:
1942
0
          case F_CR:
1943
            /*previous=crlf and now !=' '*/
1944
0
            goto endofheader;
1945
0
          case F_COMMENT:/*everything is allowed in a comment*/
1946
0
            vb->comment.s=tmp;
1947
0
            state=P_COMMENT;
1948
0
            break;
1949
0
          case P_COMMENT: /*everything is allowed in a comment*/
1950
0
            break;
1951
0
          default:
1952
0
            LM_ERR("on <%c> state %d\n",*tmp, state);
1953
0
            goto  parse_error;
1954
0
        }
1955
0
      break;
1956
0
      case '(':
1957
0
        switch(state){
1958
0
          case F_HOST:
1959
0
          case F_PORT:
1960
0
          case F_PARAM:
1961
0
          case F_VIA:
1962
0
          case F_IP6HOST:
1963
0
          case P_IP6HOST: /*must be terminated in ']'*/
1964
0
            LM_ERR(" on <%c> state %d\n", *tmp, state);
1965
0
            goto  parse_error;
1966
0
          case P_HOST:
1967
            /*mark the end*/
1968
0
            vb->host.len=tmp-vb->host.s;
1969
0
            state=F_COMMENT;
1970
0
            c_nest++;
1971
0
            break;
1972
0
          case P_PORT:
1973
            /*mark the end*/
1974
0
            vb->port_str.len=tmp-vb->port_str.s;
1975
0
            state=F_COMMENT;
1976
0
            c_nest++;
1977
0
            break;
1978
0
          case P_PARAM:
1979
            /*mark the end*/
1980
0
            vb->params.len=tmp-vb->params.s;
1981
0
            state=F_COMMENT;
1982
0
            c_nest++;
1983
0
            break;
1984
0
          case L_PORT:
1985
0
          case L_PARAM:
1986
0
          case L_VIA:
1987
0
            state=F_COMMENT;
1988
0
            vb->params.len=tmp-vb->params.s;
1989
0
            c_nest++;
1990
0
            break;
1991
0
          case P_COMMENT:
1992
0
          case F_COMMENT:
1993
0
            c_nest++;
1994
0
            break;
1995
0
          case F_CRLF:
1996
0
          case F_LF:
1997
0
          case F_CR:
1998
            /*previous=crlf and now !=' '*/
1999
0
            goto endofheader;
2000
0
          default:
2001
0
            LM_ERR("on <%c> state %d\n", *tmp, state);
2002
0
            goto  parse_error;
2003
0
        }
2004
0
      break;
2005
0
      case ')':
2006
0
        switch(state){
2007
0
          case F_COMMENT:
2008
0
          case P_COMMENT:
2009
0
            if (c_nest){
2010
0
              c_nest--;
2011
0
              if(c_nest==0){
2012
0
                state=L_VIA;
2013
0
                vb->comment.len=tmp-vb->comment.s;
2014
0
                break;
2015
0
              }
2016
0
            }else{
2017
0
              LM_ERR(" missing '(' - nesting= %d\n", c_nest);
2018
0
               goto parse_error;
2019
0
            }
2020
0
            break;
2021
0
          case F_HOST:
2022
0
          case F_PORT:
2023
0
          case F_PARAM:
2024
0
          case F_VIA:
2025
0
          case P_HOST:
2026
0
          case P_PORT:
2027
0
          case P_PARAM:
2028
0
          case L_PORT:
2029
0
          case L_PARAM:
2030
0
          case L_VIA:
2031
0
          case F_IP6HOST:
2032
0
          case P_IP6HOST:
2033
0
            LM_ERR(" on <%c> state %d\n",*tmp, state);
2034
0
            goto  parse_error;
2035
0
          case F_CRLF:
2036
0
          case F_LF:
2037
0
          case F_CR:
2038
            /*previous=crlf and now !=' '*/
2039
0
            goto endofheader;
2040
0
          default:
2041
0
            LM_ERR("on <%c> state %d\n", *tmp, state);
2042
0
            goto  parse_error;
2043
0
        }
2044
0
        break;
2045
0
      case '[':
2046
0
        switch(state){
2047
0
          case F_HOST:
2048
0
            vb->host.s=tmp; /* mark start here (include [])*/
2049
0
            state=F_IP6HOST;
2050
0
            break;
2051
0
          case F_COMMENT:/*everything is allowed in a comment*/
2052
0
            vb->comment.s=tmp;
2053
0
            state=P_COMMENT;
2054
0
            break;
2055
0
          case P_COMMENT:
2056
0
            break;
2057
0
          case F_CRLF:
2058
0
          case F_LF:
2059
0
          case F_CR:
2060
            /*previous=crlf and now !=' '*/
2061
0
            goto endofheader;
2062
0
          default:
2063
0
            LM_ERR("on <%c> state %d\n",*tmp, state);
2064
0
            goto  parse_error;
2065
0
        }
2066
0
        break;
2067
0
      case ']':
2068
0
        switch(state){
2069
0
          case P_IP6HOST:
2070
            /*mark the end*/
2071
0
            vb->host.len=(tmp-vb->host.s)+1; /* include "]" */
2072
0
            state=L_PORT;
2073
0
            break;
2074
0
          case F_CRLF:
2075
0
          case F_LF:
2076
0
          case F_CR:
2077
            /*previous=crlf and now !=' '*/
2078
0
            goto endofheader;
2079
0
          case F_COMMENT:/*everything is allowed in a comment*/
2080
0
            vb->comment.s=tmp;
2081
0
            state=P_COMMENT;
2082
0
            break;
2083
0
          case P_COMMENT:
2084
0
            break;
2085
0
          default:
2086
0
            LM_ERR("on <%c> state %d\n",*tmp, state);
2087
0
            goto  parse_error;
2088
0
        }
2089
0
        break;
2090
0
      case '\0':
2091
0
        break;
2092
2093
0
      default:
2094
0
        switch(state){
2095
0
          case F_HOST:
2096
0
            state=P_HOST;
2097
0
            vb->host.s=tmp;
2098
            //break;
2099
0
          case P_HOST:
2100
            /*check if host allowed char*/
2101
0
            if ( (*tmp<'a' || *tmp>'z') && (*tmp<'A' || *tmp>'Z')
2102
0
            && (*tmp<'0' || *tmp>'9') && *tmp!='-' && *tmp!='.')
2103
0
              goto parse_error;
2104
0
            break;
2105
0
          case F_PORT:
2106
0
            state=P_PORT;
2107
0
            vb->port_str.s=tmp;
2108
            //break;
2109
0
          case P_PORT:
2110
            /*check if number*/
2111
0
            if ( *tmp<'0' || *tmp>'9' )
2112
0
              goto parse_error;
2113
0
            break;
2114
0
          case F_PARAM:
2115
0
            /*state=P_PARAM*/;
2116
0
            if(vb->params.s==0) vb->params.s=param_start;
2117
0
            param=pkg_malloc(sizeof(struct via_param));
2118
0
            if (param==0){
2119
0
              LM_ERR("no pkg memory left\n");
2120
0
              goto error;
2121
0
            }
2122
0
            memset(param,0, sizeof(struct via_param));
2123
0
            param->start=param_start;
2124
0
            tmp=parse_via_param(tmp, end, &state, &saved_state,
2125
0
                      param);
2126
2127
0
            switch(state){
2128
0
              case F_PARAM:
2129
0
                param_start=tmp+1;
2130
0
              case L_PARAM:
2131
0
              case F_LF:
2132
0
              case F_CR:
2133
0
                vb->params.len=tmp - vb->params.s;
2134
0
                break;
2135
0
              case F_VIA:
2136
0
                vb->params.len=param->start+param->size
2137
0
                        -vb->params.s;
2138
0
                break;
2139
0
              case END_OF_HEADER:
2140
0
                vb->params.len=param->start+param->size
2141
0
                        -vb->params.s;
2142
0
                break;
2143
0
              case PARAM_ERROR:
2144
0
                pkg_free(param);
2145
0
                goto parse_error;
2146
0
              default:
2147
0
                pkg_free(param);
2148
0
                LM_ERR(" after parse_via_param: invalid "
2149
0
                    "char <%c> on state %d\n",*tmp, state);
2150
0
                goto parse_error;
2151
0
            }
2152
            /*add param to the list*/
2153
0
            if (vb->last_param) vb->last_param->next=param;
2154
0
            else        vb->param_lst=param;
2155
0
            vb->last_param=param;
2156
            /* update param. shortcuts */
2157
0
            switch(param->type){
2158
0
              case PARAM_BRANCH:
2159
0
                vb->branch=param;
2160
0
                break;
2161
0
              case PARAM_RECEIVED:
2162
0
                vb->received=param;
2163
0
                break;
2164
0
              case PARAM_RPORT:
2165
0
                vb->rport=param;
2166
0
                break;
2167
0
              case PARAM_I:
2168
0
                vb->i=param;
2169
0
                break;
2170
0
              case PARAM_ALIAS:
2171
0
                vb->alias=param;
2172
0
                break;
2173
0
              case PARAM_MADDR:
2174
0
                vb->maddr=param;
2175
0
                break;
2176
0
            }
2177
2178
0
            if (state==END_OF_HEADER){
2179
0
              state=saved_state;
2180
0
              goto endofheader;
2181
0
            }
2182
0
            break;
2183
0
          case P_PARAM:
2184
0
            break;
2185
0
          case F_VIA:
2186
            /*vb->next=tmp;*/ /*???*/
2187
0
            goto nextvia;
2188
0
          case L_PORT:
2189
0
          case L_PARAM:
2190
0
          case L_VIA:
2191
0
            LM_ERR("on <%c> state %d (default)\n",*tmp, state);
2192
0
            goto  parse_error;
2193
0
          case F_COMMENT:
2194
0
            state=P_COMMENT;
2195
0
            vb->comment.s=tmp;
2196
0
            break;
2197
0
          case P_COMMENT:
2198
0
            break;
2199
0
          case F_IP6HOST:
2200
0
            state=P_IP6HOST;
2201
            //break;
2202
0
          case P_IP6HOST:
2203
            /*check if host allowed char*/
2204
0
            if ( (*tmp<'a' || *tmp>'f') && (*tmp<'A' || *tmp>'F')
2205
0
            && (*tmp<'0' || *tmp>'9') && *tmp!=':')
2206
0
              goto parse_error;
2207
0
            break;
2208
0
          case F_CRLF:
2209
0
          case F_LF:
2210
0
          case F_CR:
2211
            /*previous=crlf and now !=' '*/
2212
0
            goto endofheader;
2213
0
          default:
2214
0
            LM_ERR("invalid char <%c> in state %d\n",*tmp, state);
2215
0
            goto parse_error;
2216
0
        }
2217
2218
2219
0
    }
2220
0
  }
2221
2222
0
  LM_DBG("end of packet reached, state=%d\n", state);
2223
0
  goto endofpacket; /*end of packet, probably should be goto error*/
2224
2225
0
endofheader:
2226
0
  state=saved_state;
2227
0
  LM_DBG("end of header reached, state=%d\n", state);
2228
0
endofpacket:
2229
  /* check if error*/
2230
0
  switch(state){
2231
0
    case P_HOST:
2232
0
    case L_PORT:
2233
0
    case P_PORT:
2234
0
    case L_PARAM:
2235
0
    case P_PARAM:
2236
0
    case P_VALUE:
2237
0
    case GEN_PARAM:
2238
0
    case FIN_HIDDEN:
2239
0
    case L_VIA:
2240
0
      break;
2241
0
    default:
2242
0
      LM_ERR(" invalid via - end of header in state %d\n", state);
2243
0
      goto parse_error;
2244
0
  }
2245
2246
2247
  /*
2248
  if (proto) printf("<SIP/2.0/%s>\n", proto);
2249
  if (host) printf("host= <%s>\n", host);
2250
  if (port_str) printf("port= <%s>\n", port_str);
2251
  if (param) printf("params= <%s>\n", param);
2252
  if (comment) printf("comment= <%s>\n", comment);
2253
  if(next_via) printf("next_via= <%s>\n", next_via);
2254
  */
2255
  /*LM_DBG("rest=<%s>\n", tmp);*/
2256
2257
0
  vb->error=PARSE_OK;
2258
0
  vb->bsize=tmp-buffer;
2259
0
  if (vb->port_str.s){
2260
0
    vb->port=str2s(vb->port_str.s, vb->port_str.len, &err);
2261
0
    if (err){
2262
0
          LM_ERR(" invalid port number <%.*s>\n",
2263
0
            vb->port_str.len, ZSW(vb->port_str.s));
2264
0
          goto parse_error;
2265
0
    }
2266
0
  }
2267
0
  return tmp;
2268
0
nextvia:
2269
0
  LM_DBG("next_via\n");
2270
0
  vb->error=PARSE_OK;
2271
0
  vb->bsize=tmp-buffer;
2272
0
  if (vb->port_str.s){
2273
0
    vb->port=str2s(vb->port_str.s, vb->port_str.len, &err);
2274
0
    if (err){
2275
0
          LM_ERR(" invalid port number <%.*s>\n",
2276
0
            vb->port_str.len, ZSW(vb->port_str.s));
2277
0
          goto parse_error;
2278
0
    }
2279
0
  }
2280
0
  vb->next=pkg_malloc(sizeof(struct via_body));
2281
0
  if (vb->next==0){
2282
0
    LM_ERR(" out of pkg memory\n");
2283
0
    goto error;
2284
0
  }
2285
0
  vb=vb->next;
2286
0
  memset(vb, 0, sizeof(struct via_body));
2287
0
  buffer=tmp;
2288
0
  goto parse_again;
2289
2290
0
parse_error:
2291
0
  if (end>buffer){
2292
0
    LM_ERR(" <%.*s>\n", (int)(end-buffer), ZSW(buffer));
2293
0
  }
2294
0
  if ((tmp>buffer)&&(tmp<end)){
2295
0
    LM_ERR("parsed so far:<%.*s>\n",
2296
0
        (int)(tmp-buffer), ZSW(buffer) );
2297
0
  }else{
2298
0
    LM_ERR("via parse failed\n");
2299
0
  }
2300
0
error:
2301
0
  vb->error=PARSE_ERROR;
2302
0
  vbody->error=PARSE_ERROR; /* make sure the first via body is marked
2303
                 as bad also */
2304
0
  return tmp;
2305
0
}
2306
2307
2308
static inline void free_via_param_list(struct via_param* vp)
2309
0
{
2310
0
  struct via_param* foo;
2311
0
  while(vp){
2312
0
    foo=vp;
2313
0
    vp=vp->next;
2314
0
    pkg_free(foo);
2315
0
  }
2316
0
}
2317
2318
2319
void free_via_list(struct via_body* vb)
2320
0
{
2321
0
  struct via_body* foo;
2322
0
  while(vb){
2323
0
    foo=vb;
2324
0
    vb=vb->next;
2325
0
    if (foo->param_lst) free_via_param_list(foo->param_lst);
2326
0
    pkg_free(foo);
2327
0
  }
2328
0
}