Coverage Report

Created: 2025-10-12 06:44

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/opensips/parser/parse_uri.c
Line
Count
Source
1
/*
2
 * Copyright (C) 2001-2003 FhG Fokus
3
 *
4
 * This file is part of opensips, a free SIP server.
5
 *
6
 * opensips is free software; you can redistribute it and/or modify
7
 * it under the terms of the GNU General Public License as published by
8
 * the Free Software Foundation; either version 2 of the License, or
9
 * (at your option) any later version
10
 *
11
 * opensips is distributed in the hope that it will be useful,
12
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14
 * GNU General Public License for more details.
15
 *
16
 * You should have received a copy of the GNU General Public License
17
 * along with this program; if not, write to the Free Software
18
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301  USA
19
 *
20
 * History:
21
 * --------
22
 * 2003-04-04  convenience inbound-uri parser parse_orig_ruri
23
 *             introduced (jiri)
24
 * 2003-04-11  new parse_uri introduced (better, parses also some parameters,
25
 *              works in one pass) (andrei)
26
 * 2003-04-11  ser_error is now set in parse_uri (andrei)
27
 * 2003-04-26  ZSW (jiri)
28
 * 2003-07-03  sips:, r2, lr=on support added (andrei)
29
 * 2005-02-25  preliminary tel uri support (andrei)
30
 * 2005-03-03  more tel uri fixes (andrei)
31
 * 2006-11-28  Added statistic support for the number of bad URI's
32
 *             (Jeffrey Magder - SOMA Networks)
33
 *  2011-04-20  added support for URI unknown parameters (osas)
34
 */
35
36
37
#include "parse_uri.h"
38
#include <string.h>
39
#include "../dprint.h"
40
#include "../ut.h"   /* q_memchr */
41
#include "../error.h"
42
#include "../errinfo.h"
43
#include "../core_stats.h"
44
#include "../strcommon.h"
45
46
static const str uri_type_names[7] = {
47
  {NULL, 0}, /*This is the error type*/
48
  str_init("sip"),
49
  str_init("sips"),
50
  str_init("tel"),
51
  str_init("tels"),
52
  str_init("urn:service"),
53
  str_init("urn:nena:service")
54
};
55
56
char* uri_type2str(const uri_type type, char *result)
57
0
{
58
0
  if (type == ERROR_URI_T)
59
0
    return NULL;
60
61
0
  memcpy(result, uri_type_names[type].s, uri_type_names[type].len);
62
0
  return result + uri_type_names[type].len;
63
0
}
64
65
int uri_typestrlen(const uri_type type)
66
0
{
67
0
  return uri_type_names[type].len;
68
0
}
69
70
uri_type str2uri_type(char * buf)
71
0
{
72
0
  int scheme = 0;
73
0
  uri_type type = ERROR_URI_T;
74
0
  scheme=buf[0]+(buf[1]<<8)+(buf[2]<<16)+(buf[3]<<24);
75
0
  scheme|=0x20202020;
76
0
  if (scheme==SIP_SCH){
77
0
    type=SIP_URI_T;
78
0
  }else if(scheme==SIPS_SCH){
79
0
    if(buf[4]==':')
80
0
      type=SIPS_URI_T;
81
0
    else type = ERROR_URI_T;
82
0
  }else if (scheme==TEL_SCH){
83
0
    type=TEL_URI_T;
84
0
  }else if (scheme==URN_SERVICE_SCH){
85
0
    if (memcmp(buf+3,URN_SERVICE_STR,URN_SERVICE_STR_LEN) == 0) {
86
0
      type=URN_SERVICE_URI_T;
87
0
    }
88
0
    else if (memcmp(buf+3,URN_NENA_SERVICE_STR,URN_NENA_SERVICE_STR_LEN) == 0) {
89
0
      type=URN_NENA_SERVICE_URI_T;
90
0
    }
91
0
  }
92
0
  return type;
93
0
}
94
95
int parse_uri_headers(str headers, str h_name[], str h_val[], int h_size)
96
0
{
97
0
  enum states {URI_H_HEADER, URI_H_VALUE};
98
0
  register enum states state;
99
0
  char* h; /* header start */
100
0
  char* v; /* header value start */
101
0
  str* header;    /* current header */
102
0
  str* header_val;  /* current header val */
103
0
  register char* p;
104
0
  char* end;
105
0
  unsigned int i = 0;
106
107
  /* init */
108
0
  end = headers.s + headers.len;
109
0
  p = h = headers.s;
110
0
  v = NULL;
111
0
  header = &h_name[0];
112
0
  header_val = &h_val[0];
113
0
  state = URI_H_HEADER;
114
0
  memset(h_name, 0, h_size * sizeof(str));
115
0
  memset(h_val, 0, h_size * sizeof(str));
116
117
0
  for(;p<end; p++){
118
0
    switch((unsigned char)state){
119
0
    case URI_H_HEADER:
120
0
      switch(*p){
121
0
      case '=':
122
0
        v = p+1;
123
0
        header->s = h;
124
0
        header->len = p-h;
125
0
        state = URI_H_VALUE;
126
0
        break;
127
0
      case '?':
128
0
        LM_ERR("Header without value\n");
129
0
        h = p+1;
130
0
        header->s = h;
131
0
        header->len = p-h;
132
0
        header_val->s = NULL;
133
0
        header_val->len = 0;
134
135
        /* advance header and header_val */
136
0
        i++;
137
0
        if(i<h_size){
138
0
          header = &h_name[i];
139
0
          header_val = &h_val[i];
140
0
        } else {
141
0
          LM_ERR("To many URI headers\n");
142
0
          return -1;
143
0
        }
144
0
        break;
145
0
      }
146
0
      break;
147
0
    case URI_H_VALUE:
148
0
      switch(*p){
149
0
      case '=':
150
0
        LM_ERR("Ignoring unexpected '=' inside URI header value\n");
151
0
        break;
152
0
      case '?':
153
0
        h = p+1;
154
0
        header_val->s = v;
155
0
        header_val->len = p-v;
156
0
        state = URI_H_HEADER;
157
158
        /* advance header and header_val */
159
0
        i++;
160
0
        if(i<h_size){
161
0
          header = &h_name[i];
162
0
          header_val = &h_val[i];
163
0
        } else {
164
0
          LM_ERR("To many URI headers\n");
165
0
          return -1;
166
0
        }
167
0
        break;
168
0
      }
169
0
      break;
170
0
    default:
171
0
      LM_ERR("Unexpected state [%d]\n", state);
172
0
      return -1;
173
0
    }
174
0
  }
175
176
0
  switch(state){
177
0
  case URI_H_HEADER:
178
0
    LM_ERR("Header without value\n");
179
0
    header->s = h;
180
0
    header->len = p-h;
181
0
    header_val->s = NULL;
182
0
    header_val->len = 0;
183
0
    break;
184
0
  case URI_H_VALUE:
185
0
    header_val->s = v;
186
0
    header_val->len = p-v;
187
0
    break;
188
0
  }
189
190
#ifdef EXTRA_DEBUG
191
  for(i=0; i<h_size && h_name[i].s; i++)
192
    LM_DBG("header=[%p]-><%.*s> val=[%p]-><%.*s>\n",
193
      h_name[i].s, h_name[i].len, h_name[i].s,
194
      h_val[i].s, h_val[i].len, h_val[i].s);
195
#endif
196
197
0
  return 0;
198
0
}
199
200
int print_uri(struct sip_uri *uri, str *out_buf)
201
0
{
202
0
#define append_str_chunk(field) \
203
0
  do { \
204
0
    if (bytes + uri->field.len > out_buf->len) { \
205
0
      LM_ERR("no more space left! printed so far: '%.*s'\n", \
206
0
               bytes, out_buf->s); \
207
0
      return -1; \
208
0
    } \
209
0
    memcpy(out_buf->s + bytes, uri->field.s, uri->field.len); \
210
0
    bytes += uri->field.len; \
211
0
  } while (0)
212
213
0
#define append_char(ch) \
214
0
  do { \
215
0
    if (bytes + 1 > out_buf->len) { \
216
0
      LM_ERR("no more space left! printed so far: '%.*s'\n", \
217
0
               bytes, out_buf->s); \
218
0
      return -1; \
219
0
    } \
220
0
    out_buf->s[bytes++] = ch; \
221
0
  } while (0)
222
223
0
#define VAL(p) p##_val
224
225
0
#define append_param(p) \
226
0
  do { \
227
0
    if (uri->p.s) { \
228
0
      append_char(';'); \
229
0
      append_str_chunk(p); \
230
0
    } \
231
0
  } while (0)
232
233
0
#define append_uk_param(idx) \
234
0
  do { \
235
0
    if (uri->u_name[idx].s) { \
236
0
      append_char(';'); \
237
0
      if (bytes + uri->u_name[idx].len > out_buf->len) { \
238
0
        LM_ERR("no more space left! printed so far: '%.*s'\n", \
239
0
                 bytes, out_buf->s); \
240
0
        return -1; \
241
0
      } \
242
0
      memcpy(out_buf->s + bytes, uri->u_name[idx].s, uri->u_name[idx].len); \
243
0
      bytes += uri->u_name[idx].len; \
244
0
      if (uri->u_val[idx].s) { \
245
0
        append_char('='); \
246
0
        if (bytes + uri->u_val[idx].len > out_buf->len) { \
247
0
          LM_ERR("no more space left! printed so far: '%.*s'\n", \
248
0
                   bytes, out_buf->s); \
249
0
          return -1; \
250
0
        } \
251
0
        memcpy(out_buf->s + bytes, uri->u_val[idx].s, uri->u_val[idx].len); \
252
0
        bytes += uri->u_val[idx].len; \
253
0
      } \
254
0
    } \
255
0
  } while (0)
256
257
0
  int bytes = 0;
258
0
  int i;
259
260
0
  memcpy(out_buf->s, uri_type_names[uri->type].s, uri_type_names[uri->type].len);
261
0
  bytes += uri_type_names[uri->type].len;
262
0
  append_char(':');
263
0
  append_str_chunk(user);
264
0
  if (uri->passwd.s) {
265
0
    append_char(':');
266
0
    append_str_chunk(passwd);
267
0
  }
268
0
  if (uri->host.s) {
269
0
    append_char('@');
270
0
    append_str_chunk(host);
271
0
  }
272
0
  if (uri->port.s) {
273
0
    append_char(':');
274
0
    append_str_chunk(port);
275
0
  }
276
277
0
  append_param(transport);
278
0
  append_param(ttl);
279
0
  append_param(user_param);
280
0
  append_param(maddr);
281
0
  append_param(method);
282
0
  append_param(lr);
283
0
  append_param(r2);
284
0
  append_param(gr);
285
0
  append_param(pn_provider);
286
0
  append_param(pn_prid);
287
0
  append_param(pn_param);
288
0
  append_param(pn_purr);
289
290
0
  for (i = 0; i < uri->u_params_no; i++)
291
0
    append_uk_param(i);
292
293
0
  out_buf->len = bytes;
294
295
0
  return 0;
296
0
#undef append_str_chunk
297
0
#undef append_char
298
0
#undef VAL
299
0
#undef append_param
300
0
#undef append_uk_param
301
0
}
302
303
/* buf= pointer to beginning of uri (sip:x@foo.bar:5060;a=b?h=i)
304
 * len= len of uri
305
 * returns: fills uri & returns <0 on error or 0 if ok
306
 */
307
int parse_uri(char* buf, int len, struct sip_uri* uri)
308
124k
{
309
124k
  enum states  {  URI_INIT, URI_USER, URI_PASSWORD, URI_PASSWORD_ALPHA,
310
124k
          URI_HOST, URI_HOST_P,
311
124k
          URI_HOST6_P, URI_HOST6_END, URI_PORT,
312
124k
          URI_PARAM, URI_PARAM_P, URI_PARAM_VAL_P,
313
124k
          URI_VAL_P, URI_HEADERS,
314
          /* param states */
315
          /* transport */
316
124k
          PT_T, PT_R, PT_A, PT_N, PT_S, PT_P, PT_O, PT_R2, PT_T2,
317
124k
          PT_eq,
318
          /* ttl */
319
124k
          PTTL_T2, PTTL_L, PTTL_eq,
320
          /* user */
321
124k
          PU_U, PU_S, PU_E, PU_R, PU_eq,
322
          /* method */
323
124k
          PM_M, PM_E, PM_T, PM_H, PM_O, PM_D, PM_eq,
324
          /* maddr */
325
124k
          PMA_A, PMA_D, PMA_D2, PMA_R, PMA_eq,
326
          /* lr */
327
124k
          PLR_L, PLR_R_FIN, PLR_eq,
328
          /* gr */
329
124k
          PG_G, PG_G_FIN, PG_eq,
330
          /* r2 */
331
124k
          PR2_R, PR2_2_FIN, PR2_eq,
332
          /* transport values */
333
          /* udp */
334
124k
          VU_U, VU_D, VU_P_FIN,
335
          /* tcp */
336
124k
          VT_T, VT_C, VT_P_FIN,
337
          /* tls */
338
124k
          VTLS_L, VTLS_S_FIN,
339
          /* sctp */
340
124k
          VS_S, VS_C, VS_T, VS_P_FIN,
341
          /* ws */
342
124k
          VW_W, VW_S, VW_S_FIN, VWS_S_FIN,
343
344
          /* pn-{provider, prid, param, purr} (RFC 8599 - SIP PN) */
345
124k
          PN_P, PN_N, PN_dash, PN_P2, PN_PR,
346
124k
          PN1_O, PN1_V, PN1_I, PN1_D, PN1_E, PN1_FIN, PN1_eq,
347
124k
          PN2_I, PN2_D, PN2_eq,
348
124k
          PN3_A, PN3_R, PN3_A2, PN3_M, PN3_eq,
349
124k
          PN4_U, PN4_R, PN4_R2, PN4_eq,
350
351
124k
  };
352
124k
  register enum states state;
353
124k
  char* s;
354
124k
  char* b; /* param start */
355
124k
  char *v; /* value start */
356
124k
  str* param; /* current param */
357
124k
  str* param_val; /* current param val */
358
124k
  str user;
359
124k
  str password;
360
124k
  unsigned int port_no;
361
124k
  register char* p;
362
124k
  char* end;
363
124k
  char* pass;
364
124k
  int found_user;
365
124k
  int error_headers;
366
124k
  unsigned int scheme;
367
124k
  uri_type backup;
368
#ifdef EXTRA_DEBUG
369
  int i;
370
#endif
371
372
124k
#define case_port( ch, var, ovf_check1, ovf_check2) \
373
124k
  case ch: \
374
54.8k
      if (ovf_check1) \
375
54.8k
        (var)=(var)*10+(ch-'0'); \
376
54.8k
      if (ovf_check2 && (var) > USHRT_MAX) \
377
54.8k
        goto error_bad_port; \
378
54.8k
      break
379
380
124k
#define still_at_user  \
381
124k
            if (found_user==0){ \
382
24.2k
              user.s=uri->host.s; \
383
24.2k
              if (pass){\
384
11.0k
                user.len=pass-user.s; \
385
11.0k
                password.s=pass+1; \
386
11.0k
                password.len=p-password.s; \
387
13.2k
              }else{ \
388
13.2k
                user.len=p-user.s; \
389
13.2k
              }\
390
              /* save the uri type/scheme */ \
391
24.2k
              backup=uri->type; \
392
              /* everything else is 0 */ \
393
24.2k
              memset(uri, 0, sizeof(struct sip_uri)); \
394
              /* restore the scheme, copy user & pass */ \
395
24.2k
              uri->type=backup; \
396
24.2k
              uri->user=user; \
397
24.2k
              if (pass) uri->passwd=password;  \
398
24.2k
              s=p+1; \
399
24.2k
              found_user=1;\
400
24.2k
              error_headers=0; \
401
24.2k
              state=URI_HOST; \
402
24.2k
            }else goto error_bad_char
403
404
124k
#define check_host_end \
405
124k
          case ':': \
406
            /* found the host */ \
407
2.60k
            uri->host.s=s; \
408
2.60k
            uri->host.len=p-s; \
409
2.60k
            state=URI_PORT; \
410
2.60k
            s=p+1; \
411
2.60k
            break; \
412
13.5k
          case ';': \
413
13.5k
            uri->host.s=s; \
414
13.5k
            uri->host.len=p-s; \
415
13.5k
            state=URI_PARAM; \
416
13.5k
            s=p+1; \
417
13.5k
            break; \
418
13.5k
          case '?': \
419
235
            uri->host.s=s; \
420
235
            uri->host.len=p-s; \
421
235
            state=URI_HEADERS; \
422
235
            s=p+1; \
423
235
            break; \
424
235
          case '&': \
425
410
          case '@': \
426
410
            goto error_bad_char
427
428
429
124k
#define param_set(t_start, v_start) \
430
144k
          param->s=(t_start);\
431
144k
          param->len=(p-(t_start));\
432
144k
          param_val->s=(v_start); \
433
144k
          param_val->len=(p-(v_start))
434
435
124k
#define u_param_set(t_start, v_start) \
436
556k
      if (uri->u_params_no < URI_MAX_U_PARAMS){ \
437
140k
        if((v_start)>(t_start)){ \
438
10.8k
          uri->u_name[uri->u_params_no].s=(t_start); \
439
10.8k
          uri->u_name[uri->u_params_no].len=((v_start)-(t_start)-1); \
440
10.8k
          if(p>(v_start)) { \
441
8.28k
            uri->u_val[uri->u_params_no].s=(v_start); \
442
8.28k
            uri->u_val[uri->u_params_no].len=(p-(v_start)); \
443
8.28k
          } \
444
129k
        } else { \
445
129k
          uri->u_name[uri->u_params_no].s=(t_start); \
446
129k
          uri->u_name[uri->u_params_no].len=(p-(t_start)); \
447
129k
        } \
448
140k
        uri->u_params_no++; \
449
416k
      } else { \
450
416k
        LM_ERR("unknown URI param list excedeed\n"); \
451
416k
      }
452
453
124k
#define semicolon_case \
454
852k
          case';': \
455
852k
            if (pass){ \
456
10.8k
              found_user=1;/* no user, pass cannot contain ';'*/ \
457
10.8k
              pass=0; \
458
10.8k
            } \
459
852k
            state=URI_PARAM   /* new param */
460
461
124k
#define question_case \
462
124k
          case '?': \
463
16.8k
            uri->params.s=s; \
464
16.8k
            uri->params.len=p-s; \
465
16.8k
            state=URI_HEADERS; \
466
16.8k
            s=p+1; \
467
16.8k
            if (pass){ \
468
3.87k
              found_user=1;/* no user, pass cannot contain '?'*/ \
469
3.87k
              pass=0; \
470
3.87k
            }
471
472
124k
#define colon_case \
473
287k
          case ':': \
474
287k
            if (found_user==0){ \
475
              /*might be pass but only if user not found yet*/ \
476
19.7k
              if (pass){ \
477
1.42k
                found_user=1; /* no user */ \
478
1.42k
                pass=0; \
479
18.3k
              }else{ \
480
18.3k
                pass=p; \
481
18.3k
              } \
482
19.7k
            } \
483
287k
            state=URI_PARAM_P /* generic param */
484
485
124k
#define param_common_cases \
486
124k
          case '@': \
487
            /* ughhh, this is still the user */ \
488
6.03k
            still_at_user; \
489
3.26k
            break; \
490
164k
          semicolon_case; \
491
164k
            break; \
492
164k
          question_case; \
493
3.18k
            break; \
494
45.4k
          colon_case; \
495
45.4k
            break
496
497
124k
#define u_param_common_cases \
498
124k
          case '@': \
499
            /* ughhh, this is still the user */ \
500
24.2k
            still_at_user; \
501
14.1k
            break; \
502
541k
          semicolon_case; \
503
541k
            u_param_set(b, v); \
504
541k
            break; \
505
541k
          question_case; \
506
10.7k
            u_param_set(b, v); \
507
10.7k
            break; \
508
186k
          colon_case; \
509
186k
            break
510
511
124k
#define value_common_cases \
512
124k
          case '@': \
513
            /* ughhh, this is still the user */ \
514
3.06k
            still_at_user; \
515
1.43k
            break; \
516
132k
          semicolon_case; \
517
132k
            param_set(b, v); \
518
132k
            break; \
519
132k
          question_case; \
520
1.66k
            param_set(b, v); \
521
1.66k
            break; \
522
40.7k
          colon_case; \
523
40.7k
            state=URI_VAL_P; \
524
40.7k
            break
525
526
124k
#define param_switch(old_state, c1, c2, new_state) \
527
2.19M
      case old_state: \
528
2.19M
        switch(*p){ \
529
1.13M
          case c1: \
530
1.91M
          case c2: \
531
1.91M
            state=(new_state); \
532
1.91M
            break; \
533
1.13M
          u_param_common_cases; \
534
143k
          default: \
535
143k
            state=URI_PARAM_P; \
536
2.19M
        } \
537
2.19M
        break
538
124k
#define param_switch1(old_state, c1, new_state) \
539
376k
      case old_state: \
540
376k
        switch(*p){ \
541
296k
          case c1: \
542
296k
            state=(new_state); \
543
296k
            break; \
544
770
          param_common_cases; \
545
28.1k
          default: \
546
28.1k
            state=URI_PARAM_P; \
547
376k
        } \
548
376k
        break
549
124k
#define param_xswitch1(old_state, c1, new_state) \
550
124k
      case old_state: \
551
15.5k
        switch(*p){ \
552
15.3k
          case c1: \
553
15.3k
            state=(new_state); \
554
15.3k
            break; \
555
220
          default: \
556
220
            goto error_bad_char; \
557
15.5k
        } \
558
15.5k
        break
559
124k
#define param_switch_big(old_state, c1, c2, d1, d2, new_state_c, new_state_d) \
560
454k
      case old_state : \
561
454k
        switch(*p){ \
562
250k
          case c1: \
563
308k
          case c2: \
564
308k
            state=(new_state_c); \
565
308k
            break; \
566
250k
          case d1: \
567
74.9k
          case d2: \
568
74.9k
            state=(new_state_d); \
569
74.9k
            break; \
570
60.7k
          u_param_common_cases; \
571
39.5k
          default: \
572
39.5k
            state=URI_PARAM_P; \
573
454k
        } \
574
454k
        break
575
124k
#define param_switch_bigger(old_state, c1, c2, d1, d2, e1, e2, new_state_c, new_state_d, new_state_e) \
576
128k
      case old_state : \
577
128k
        switch(*p){ \
578
50.3k
          case c1: \
579
76.4k
          case c2: \
580
76.4k
            state=(new_state_c); \
581
76.4k
            break; \
582
50.3k
          case d1: \
583
20.8k
          case d2: \
584
20.8k
            state=(new_state_d); \
585
20.8k
            break; \
586
5.09k
          case e1: \
587
21.5k
          case e2: \
588
21.5k
            state=(new_state_e); \
589
21.5k
            break; \
590
5.09k
          u_param_common_cases; \
591
5.47k
          default: \
592
5.47k
            state=URI_PARAM_P; \
593
128k
        } \
594
128k
        break
595
124k
#define value_switch(old_state, c1, c2, new_state) \
596
140k
      case old_state: \
597
140k
        switch(*p){ \
598
37.6k
          case c1: \
599
65.0k
          case c2: \
600
65.0k
            state=(new_state); \
601
65.0k
            break; \
602
37.6k
          value_common_cases; \
603
15.2k
          default: \
604
15.2k
            state=URI_VAL_P; \
605
140k
        } \
606
140k
        break
607
124k
#define value_switch_big(old_state, c1, c2, d1, d2, new_state_c, new_state_d) \
608
124k
      case old_state: \
609
49.3k
        switch(*p){ \
610
19.3k
          case c1: \
611
23.2k
          case c2: \
612
23.2k
            state=(new_state_c); \
613
23.2k
            break; \
614
19.3k
          case d1: \
615
22.3k
          case d2: \
616
22.3k
            state=(new_state_d); \
617
22.3k
            break; \
618
13.8k
          value_common_cases; \
619
2.18k
          default: \
620
2.18k
            state=URI_VAL_P; \
621
49.3k
        } \
622
49.3k
        break
623
624
124k
#define transport_fin(c_state, proto_no) \
625
124k
      case c_state: \
626
36.8k
        switch(*p){ \
627
527
          case '@': \
628
527
            still_at_user; \
629
527
            break; \
630
8.33k
          semicolon_case; \
631
8.33k
            param_set(b, v); \
632
8.33k
            uri->proto=(proto_no); \
633
8.33k
            break; \
634
769
          question_case; \
635
769
            param_set(b, v); \
636
769
            uri->proto=(proto_no); \
637
769
            break; \
638
11.9k
          colon_case;  \
639
27.1k
          default: \
640
27.1k
            state=URI_VAL_P; \
641
27.1k
            break; \
642
36.8k
        } \
643
36.8k
        break
644
645
646
647
  /* init */
648
124k
  end=buf+len;
649
124k
  p=buf+4;
650
124k
  found_user=0;
651
124k
  error_headers=0;
652
124k
  b=v=0;
653
124k
  param=param_val=0;
654
124k
  pass=0;
655
124k
  password.s = 0;
656
124k
  password.len = 0;
657
124k
  port_no=0;
658
124k
  state=URI_INIT;
659
124k
  memset(uri, 0, sizeof(struct sip_uri)); /* zero it all, just to be sure*/
660
  /*look for sip:, sips: or tel:*/
661
124k
  if (len<5) goto error_too_short;
662
59.4k
  scheme=(unsigned)(unsigned char)buf[0]
663
59.4k
      + (((unsigned)(unsigned char)buf[1])<<8)
664
59.4k
      + (((unsigned)(unsigned char)buf[2])<<16)
665
59.4k
      + (((unsigned)(unsigned char)buf[3])<<24);
666
59.4k
  scheme|=0x20202020;
667
59.4k
  if (scheme==SIP_SCH){
668
49.1k
    uri->type=SIP_URI_T;
669
49.1k
  }else if(scheme==SIPS_SCH){
670
327
    if(buf[4]==':'){ p++; uri->type=SIPS_URI_T;}
671
44
    else goto error_bad_uri;
672
10.0k
  }else if (scheme==TEL_SCH){
673
4.30k
    uri->type=TEL_URI_T;
674
5.70k
  }else if (scheme==URN_SERVICE_SCH){
675
1.06k
    if ((end-(buf+3)) >= URN_SERVICE_STR_LEN
676
969
            && memcmp(buf+3,URN_SERVICE_STR,URN_SERVICE_STR_LEN) == 0) {
677
26
      p+= URN_SERVICE_STR_LEN-1;
678
26
      uri->type=URN_SERVICE_URI_T;
679
26
    }
680
1.03k
    else if ((end-(buf+3)) >= URN_NENA_SERVICE_STR_LEN
681
500
            && memcmp(buf+3,URN_NENA_SERVICE_STR,URN_NENA_SERVICE_STR_LEN) == 0) {
682
80
      p+= URN_NENA_SERVICE_STR_LEN-1;
683
80
      uri->type=URN_NENA_SERVICE_URI_T;
684
954
    }else goto error_bad_uri;
685
4.64k
  }else goto error_bad_uri;
686
687
53.8k
  s=p;
688
9.92M
  for(;p<end; p++){
689
9.89M
    switch((unsigned char)state){
690
53.7k
      case URI_INIT:
691
53.7k
        switch(*p){
692
1.30k
          case '[':
693
            /* uri =  [ipv6address]... */
694
1.30k
            state=URI_HOST6_P;
695
1.30k
            s=p;
696
1.30k
            break;
697
39
          case ']':
698
            /* invalid, no uri can start with ']' */
699
111
          case ':':
700
            /* the same as above for ':' */
701
111
            goto error_bad_char;
702
53
          case '@': /* error no user part */
703
53
            goto error_bad_char;
704
52.2k
          default:
705
52.2k
            state=URI_USER;
706
53.7k
        }
707
53.5k
        break;
708
152k
      case URI_USER:
709
152k
        switch(*p){
710
1.65k
          case '@':
711
            /* found the user*/
712
1.65k
            uri->user.s=s;
713
1.65k
            uri->user.len=p-s;
714
1.65k
            state=URI_HOST;
715
1.65k
            found_user=1;
716
1.65k
            s=p+1; /* skip '@' */
717
1.65k
            break;
718
9.96k
          case ':':
719
            /* found the user, or the host? */
720
9.96k
            uri->user.s=s;
721
9.96k
            uri->user.len=p-s;
722
9.96k
            state=URI_PASSWORD;
723
9.96k
            s=p+1; /* skip ':' */
724
9.96k
            break;
725
39.1k
          case ';':
726
            /* this could be still the user or
727
             * params?*/
728
39.1k
            uri->host.s=s;
729
39.1k
            uri->host.len=p-s;
730
39.1k
            state=URI_PARAM;
731
39.1k
            s=p+1;
732
39.1k
            break;
733
1.27k
          case '?': /* still user or headers? */
734
1.27k
            uri->host.s=s;
735
1.27k
            uri->host.len=p-s;
736
1.27k
            state=URI_HEADERS;
737
1.27k
            s=p+1;
738
1.27k
            break;
739
            /* almost anything permitted in the user part */
740
43
          case '[':
741
82
          case ']': /* the user part cannot contain "[]" */
742
82
            goto error_bad_char;
743
152k
        }
744
152k
        break;
745
152k
      case URI_PASSWORD: /* this can also be the port (missing user)*/
746
52.9k
        switch(*p){
747
603
          case '@':
748
            /* found the password*/
749
603
            uri->passwd.s=s;
750
603
            uri->passwd.len=p-s;
751
603
            port_no=0;
752
603
            state=URI_HOST;
753
603
            found_user=1;
754
603
            s=p+1; /* skip '@' */
755
603
            break;
756
4.99k
          case ';':
757
            /* upps this is the port */
758
4.99k
            uri->port.s=s;
759
4.99k
            uri->port.len=p-s;
760
4.99k
            uri->port_no=port_no;
761
            /* user contains in fact the host */
762
4.99k
            uri->host.s=uri->user.s;
763
4.99k
            uri->host.len=uri->user.len;
764
4.99k
            uri->user.s=0;
765
4.99k
            uri->user.len=0;
766
4.99k
            state=URI_PARAM;
767
4.99k
            found_user=1; /*  there is no user part */
768
4.99k
            s=p+1;
769
4.99k
            break;
770
108
          case '?':
771
            /* upps this is the port */
772
108
            uri->port.s=s;
773
108
            uri->port.len=p-s;
774
108
            uri->port_no=port_no;
775
            /* user contains in fact the host */
776
108
            uri->host.s=uri->user.s;
777
108
            uri->host.len=uri->user.len;
778
108
            uri->user.s=0;
779
108
            uri->user.len=0;
780
108
            state=URI_HEADERS;
781
108
            found_user=1; /*  there is no user part */
782
108
            s=p+1;
783
108
            break;
784
10.3k
          case_port('0', port_no, port_no < INT_MAX / 10, 0);
785
10.3k
          case_port('1', port_no, port_no < INT_MAX / 10, 0);
786
4.56k
          case_port('2', port_no, port_no < INT_MAX / 10, 0);
787
4.56k
          case_port('3', port_no, port_no < INT_MAX / 10, 0);
788
5.36k
          case_port('4', port_no, port_no < INT_MAX / 10, 0);
789
5.36k
          case_port('5', port_no, port_no < INT_MAX / 10, 0);
790
4.34k
          case_port('6', port_no, port_no < INT_MAX / 10, 0);
791
3.61k
          case_port('7', port_no, port_no < INT_MAX / 10, 0);
792
3.86k
          case_port('8', port_no, port_no < INT_MAX / 10, 0);
793
3.86k
          case_port('9', port_no, port_no < INT_MAX / 10, 0);
794
3.16k
          case '[':
795
121
          case ']':
796
183
          case ':':
797
183
            goto error_bad_char;
798
1.28k
          default:
799
            /* it can't be the port, non number found */
800
1.28k
            port_no=0;
801
1.28k
            state=URI_PASSWORD_ALPHA;
802
52.9k
        }
803
52.7k
        break;
804
52.7k
      case URI_PASSWORD_ALPHA:
805
9.40k
        switch(*p){
806
196
          case '@':
807
            /* found the password*/
808
196
            uri->passwd.s=s;
809
196
            uri->passwd.len=p-s;
810
196
            state=URI_HOST;
811
196
            found_user=1;
812
196
            s=p+1; /* skip '@' */
813
196
            break;
814
225
          case ';': /* contains non-numbers => cannot be port no*/
815
292
          case '?':
816
292
            goto error_bad_port;
817
54
          case '[':
818
107
          case ']':
819
248
          case ':':
820
248
            goto error_bad_char;
821
9.40k
        }
822
8.86k
        break;
823
17.3k
      case URI_HOST:
824
17.3k
        switch(*p){
825
149
          case '[':
826
149
            state=URI_HOST6_P;
827
149
            break;
828
199
          case ':':
829
444
          case ';':
830
591
          case '?': /* null host name ->invalid */
831
658
          case '&':
832
889
          case '@': /*chars not allowed in hosts names */
833
889
            goto error_bad_host;
834
16.2k
          default:
835
16.2k
            state=URI_HOST_P;
836
17.3k
        }
837
16.4k
        break;
838
54.9k
      case URI_HOST_P:
839
54.9k
        switch(*p){
840
54.9k
          check_host_end;
841
54.9k
        }
842
54.6k
        break;
843
54.6k
      case URI_HOST6_END:
844
1.07k
        switch(*p){
845
795
          check_host_end;
846
43
          default: /*no chars allowed after [ipv6] */
847
43
            goto error_bad_host;
848
1.07k
        }
849
955
        break;
850
2.08k
      case URI_HOST6_P:
851
2.08k
        switch(*p){
852
1.10k
          case ']':
853
1.10k
            state=URI_HOST6_END;
854
1.10k
            break;
855
39
          case '[':
856
78
          case '&':
857
117
          case '@':
858
160
          case ';':
859
227
          case '?':
860
227
            goto error_bad_host;
861
2.08k
        }
862
1.85k
        break;
863
10.4k
      case URI_PORT:
864
10.4k
        switch(*p){
865
389
          case ';':
866
389
            uri->port.s=s;
867
389
            uri->port.len=p-s;
868
389
            uri->port_no=port_no;
869
389
            state=URI_PARAM;
870
389
            s=p+1;
871
389
            break;
872
55
          case '?':
873
55
            uri->port.s=s;
874
55
            uri->port.len=p-s;
875
55
            uri->port_no=port_no;
876
55
            state=URI_HEADERS;
877
55
            s=p+1;
878
55
            break;
879
1.79k
          case_port('0', port_no, 1, 1);
880
1.69k
          case_port('1', port_no, 1, 1);
881
746
          case_port('2', port_no, 1, 1);
882
595
          case_port('3', port_no, 1, 1);
883
754
          case_port('4', port_no, 1, 1);
884
1.36k
          case_port('5', port_no, 1, 1);
885
1.27k
          case_port('6', port_no, 1, 1);
886
973
          case_port('7', port_no, 1, 1);
887
697
          case_port('8', port_no, 1, 1);
888
752
          case_port('9', port_no, 1, 1);
889
655
          case '&':
890
102
          case '@':
891
158
          case ':':
892
1.01k
          default:
893
1.01k
            goto error_bad_port;
894
10.4k
        }
895
8.67k
        break;
896
908k
      case URI_PARAM: /* beginning of a new param */
897
908k
        switch(*p){
898
4.24k
          param_common_cases;
899
          /* recognized params */
900
227k
          case 't':
901
269k
          case 'T':
902
269k
            b=p;
903
269k
            state=PT_T;
904
269k
            break;
905
9.94k
          case 'u':
906
28.9k
          case 'U':
907
28.9k
            b=p;
908
28.9k
            state=PU_U;
909
28.9k
            break;
910
38.2k
          case 'm':
911
107k
          case 'M':
912
107k
            b=p;
913
107k
            state=PM_M;
914
107k
            break;
915
19.1k
          case 'l':
916
23.4k
          case 'L':
917
23.4k
            b=p;
918
23.4k
            state=PLR_L;
919
23.4k
            break;
920
14.7k
          case 'g':
921
18.2k
          case 'G':
922
18.2k
            b=p;
923
18.2k
            state=PG_G;
924
18.2k
            break;
925
34.5k
          case 'r':
926
41.1k
          case 'R':
927
41.1k
            b=p;
928
41.1k
            state=PR2_R;
929
41.1k
            break;
930
161k
          case 'p':
931
173k
          case 'P':
932
173k
            b=p;
933
173k
            state=PN_P;
934
173k
            break;
935
132k
          default:
936
132k
            b=p;
937
132k
            state=URI_PARAM_P;
938
908k
        }
939
906k
        break;
940
3.53M
      case URI_PARAM_P: /* ignore current param */
941
        /* supported params:
942
         *  maddr, transport, ttl, lr, user, method, r2  */
943
3.53M
        switch(*p){
944
19.1k
          u_param_common_cases;
945
146k
          case '=':
946
146k
            v=p + 1;
947
146k
            state=URI_PARAM_VAL_P;
948
146k
            break;
949
3.53M
        };
950
3.52M
        break;
951
941k
      case URI_PARAM_VAL_P: /* value of the ignored current param */
952
941k
        switch(*p){
953
941k
          u_param_common_cases;
954
941k
        };
955
941k
        break;
956
      /* ugly but fast param names parsing */
957
      /*transport */
958
269k
      param_switch_big(PT_T,  'r', 'R', 't', 'T', PT_R, PTTL_T2);
959
269k
      param_switch(PT_R,  'a', 'A', PT_A);
960
222k
      param_switch(PT_A,  'n', 'N', PT_N);
961
203k
      param_switch(PT_N,  's', 'S', PT_S);
962
186k
      param_switch(PT_S,  'p', 'P', PT_P);
963
176k
      param_switch(PT_P,  'o', 'O', PT_O);
964
163k
      param_switch(PT_O,  'r', 'R', PT_R2);
965
155k
      param_switch(PT_R2, 't', 'T', PT_T2);
966
146k
      param_switch1(PT_T2, '=',  PT_eq);
967
      /* value parsing */
968
139k
      case PT_eq:
969
124k
        param=&uri->transport;
970
124k
        param_val=&uri->transport_val;
971
124k
        uri->proto = PROTO_OTHER;
972
124k
        switch (*p){
973
82
          param_common_cases;
974
21.5k
          case 'u':
975
27.3k
          case 'U':
976
27.3k
            v=p;
977
27.3k
            state=VU_U;
978
27.3k
            break;
979
44.1k
          case 't':
980
49.3k
          case 'T':
981
49.3k
            v=p;
982
49.3k
            state=VT_T;
983
49.3k
            break;
984
10.5k
          case 's':
985
19.4k
          case 'S':
986
19.4k
            v=p;
987
19.4k
            state=VS_S;
988
19.4k
            break;
989
6.66k
          case 'w':
990
20.0k
          case 'W':
991
20.0k
            v=p;
992
20.0k
            state=VW_W;
993
20.0k
            break;
994
5.86k
          default:
995
5.86k
            v=p;
996
5.86k
            state=URI_VAL_P;
997
124k
        }
998
124k
        break;
999
        /* generic value */
1000
412k
      case URI_VAL_P:
1001
412k
        switch(*p){
1002
412k
          value_common_cases;
1003
412k
        }
1004
411k
        break;
1005
      /* udp */
1006
411k
      value_switch(VU_U,  'd', 'D', VU_D);
1007
27.3k
      value_switch(VU_D,  'p', 'P', VU_P_FIN);
1008
7.87k
      transport_fin(VU_P_FIN, PROTO_UDP);
1009
      /* tcp */
1010
49.3k
      value_switch_big(VT_T,  'c', 'C', 'l', 'L', VT_C, VTLS_L);
1011
49.3k
      value_switch(VT_C,  'p', 'P', VT_P_FIN);
1012
23.1k
      transport_fin(VT_P_FIN, PROTO_TCP);
1013
      /* tls */
1014
22.3k
      value_switch(VTLS_L, 's', 'S', VTLS_S_FIN);
1015
22.3k
      transport_fin(VTLS_S_FIN, PROTO_TLS);
1016
      /* sctp */
1017
19.4k
      value_switch(VS_S, 'c', 'C', VS_C);
1018
19.4k
      value_switch(VS_C, 't', 'T', VS_T);
1019
14.3k
      value_switch(VS_T, 'p', 'P', VS_P_FIN);
1020
5.74k
      transport_fin(VS_P_FIN, PROTO_SCTP);
1021
      /* ws */
1022
20.0k
      value_switch(VW_W, 's', 'S', VW_S);
1023
19.9k
      case VW_S:
1024
13.1k
        if (*p == 's' || *p == 'S') {
1025
4.36k
          state=(VWS_S_FIN);
1026
4.36k
          break;
1027
4.36k
        }
1028
        /* if not a 's' transiting to VWS_S_FIN, fallback
1029
         * to testing as existing VW_S_FIN (NOTE the missing break) */
1030
8.83k
        state=(VW_S_FIN);
1031
8.83k
      transport_fin(VW_S_FIN, PROTO_WS);
1032
8.79k
      transport_fin(VWS_S_FIN, PROTO_WSS);
1033
1034
      /* ttl */
1035
15.8k
      param_switch(PTTL_T2,  'l', 'L', PTTL_L);
1036
15.7k
      param_switch1(PTTL_L,  '=', PTTL_eq);
1037
5.61k
      case PTTL_eq:
1038
2.26k
        param=&uri->ttl;
1039
2.26k
        param_val=&uri->ttl_val;
1040
2.26k
        switch(*p){
1041
83
          param_common_cases;
1042
896
          default:
1043
896
            v=p;
1044
896
            state=URI_VAL_P;
1045
2.26k
        }
1046
2.22k
        break;
1047
1048
      /* user param */
1049
28.8k
      param_switch(PU_U, 's', 'S', PU_S);
1050
28.7k
      param_switch(PU_S, 'e', 'E', PU_E);
1051
20.0k
      param_switch(PU_E, 'r', 'R', PU_R);
1052
14.6k
      param_switch1(PU_R, '=', PU_eq);
1053
6.46k
      case PU_eq:
1054
2.17k
        param=&uri->user_param;
1055
2.17k
        param_val=&uri->user_param_val;
1056
2.17k
        switch(*p){
1057
82
          param_common_cases;
1058
727
          default:
1059
727
            v=p;
1060
727
            state=URI_VAL_P;
1061
2.17k
        }
1062
2.13k
        break;
1063
1064
      /* method*/
1065
107k
      param_switch_big(PM_M, 'e', 'E', 'a', 'A', PM_E, PMA_A);
1066
107k
      param_switch(PM_E, 't', 'T', PM_T);
1067
33.5k
      param_switch(PM_T, 'h', 'H', PM_H);
1068
28.3k
      param_switch(PM_H, 'o', 'O', PM_O);
1069
22.2k
      param_switch(PM_O, 'd', 'D', PM_D);
1070
11.7k
      param_switch1(PM_D, '=', PM_eq);
1071
4.81k
      case PM_eq:
1072
2.10k
        param=&uri->method;
1073
2.10k
        param_val=&uri->method_val;
1074
2.10k
        switch(*p){
1075
83
          param_common_cases;
1076
537
          default:
1077
537
            v=p;
1078
537
            state=URI_VAL_P;
1079
2.10k
        }
1080
2.07k
        break;
1081
1082
      /*maddr*/
1083
43.9k
      param_switch(PMA_A,  'd', 'D', PMA_D);
1084
43.9k
      param_switch(PMA_D,  'd', 'D', PMA_D2);
1085
37.6k
      param_switch(PMA_D2, 'r', 'R', PMA_R);
1086
28.9k
      param_switch1(PMA_R, '=', PMA_eq);
1087
24.7k
      case PMA_eq:
1088
15.6k
        param=&uri->maddr;
1089
15.6k
        param_val=&uri->maddr_val;
1090
15.6k
        switch(*p){
1091
83
          param_common_cases;
1092
2.25k
          default:
1093
2.25k
            v=p;
1094
2.25k
            state=URI_VAL_P;
1095
15.6k
        }
1096
15.6k
        break;
1097
1098
      /* lr */
1099
23.3k
      param_switch(PLR_L,  'r', 'R', PLR_R_FIN);
1100
23.3k
      case PLR_R_FIN:
1101
11.3k
        switch(*p){
1102
101
          case '@':
1103
101
            still_at_user;
1104
58
            break;
1105
7.08k
          case '=':
1106
7.08k
            state=PLR_eq;
1107
7.08k
            break;
1108
1.98k
          semicolon_case;
1109
1.98k
            uri->lr.s=b;
1110
1.98k
            uri->lr.len=(p-b);
1111
1.98k
            break;
1112
120
          question_case;
1113
120
            uri->lr.s=b;
1114
120
            uri->lr.len=(p-b);
1115
120
            break;
1116
835
          colon_case;
1117
835
            break;
1118
1.18k
          default:
1119
1.18k
            state=URI_PARAM_P;
1120
11.3k
        }
1121
11.2k
        break;
1122
        /* handle lr=something case */
1123
11.2k
      case PLR_eq:
1124
7.06k
        param=&uri->lr;
1125
7.06k
        param_val=&uri->lr_val;
1126
7.06k
        switch(*p){
1127
82
          param_common_cases;
1128
1.21k
          default:
1129
1.21k
            v=p;
1130
1.21k
            state=URI_VAL_P;
1131
7.06k
        }
1132
7.02k
        break;
1133
1134
      /* r2 */
1135
41.0k
      param_switch1(PR2_R,  '2', PR2_2_FIN);
1136
40.9k
      case PR2_2_FIN:
1137
7.68k
        switch(*p){
1138
82
          case '@':
1139
82
            still_at_user;
1140
43
            break;
1141
4.79k
          case '=':
1142
4.79k
            state=PR2_eq;
1143
4.79k
            break;
1144
913
          semicolon_case;
1145
913
            uri->r2.s=b;
1146
913
            uri->r2.len=(p-b);
1147
913
            break;
1148
134
          question_case;
1149
134
            uri->r2.s=b;
1150
134
            uri->r2.len=(p-b);
1151
134
            break;
1152
1.11k
          colon_case;
1153
1.11k
            break;
1154
640
          default:
1155
640
            state=URI_PARAM_P;
1156
7.68k
        }
1157
7.64k
        break;
1158
        /* handle r2=something case */
1159
7.64k
      case PR2_eq:
1160
4.77k
        param=&uri->r2;
1161
4.77k
        param_val=&uri->r2_val;
1162
4.77k
        switch(*p){
1163
128
          param_common_cases;
1164
1.31k
          default:
1165
1.31k
            v=p;
1166
1.31k
            state=URI_VAL_P;
1167
4.77k
        }
1168
4.69k
        break;
1169
1170
1171
      /* gr */
1172
18.2k
      param_switch(PG_G,  'r', 'R', PG_G_FIN);
1173
18.2k
      case PG_G_FIN:
1174
7.84k
        switch(*p){
1175
114
          case '@':
1176
114
            still_at_user;
1177
71
            break;
1178
4.42k
          case '=':
1179
4.42k
            state=PG_eq;
1180
4.42k
            break;
1181
788
          semicolon_case;
1182
788
            uri->gr.s=b;
1183
788
            uri->gr.len=(p-b);
1184
788
            break;
1185
130
          question_case;
1186
130
            uri->gr.s=b;
1187
130
            uri->gr.len=(p-b);
1188
130
            break;
1189
995
          colon_case;
1190
995
            break;
1191
1.39k
          default:
1192
1.39k
            state=URI_PARAM_P;
1193
7.84k
        }
1194
7.80k
        break;
1195
        /* handle gr=something case */
1196
7.80k
      case PG_eq:
1197
4.41k
        param=&uri->gr;
1198
4.41k
        param_val=&uri->gr_val;
1199
4.41k
        switch(*p){
1200
110
          param_common_cases;
1201
1.06k
          default:
1202
1.06k
            v=p;
1203
1.06k
            state=URI_VAL_P;
1204
4.41k
        }
1205
4.37k
        break;
1206
1207
1208
      /* pn-* */
1209
173k
      param_switch(PN_P, 'n', 'N', PN_N);
1210
173k
      param_switch1(PN_N, '-', PN_dash);
1211
154k
      param_switch(PN_dash, 'p', 'P', PN_P2);
1212
1213
141k
      param_switch_bigger(PN_P2, 'r', 'R', 'a', 'A', 'u', 'U',
1214
128k
                          PN_PR, PN3_A, PN4_U);
1215
128k
      param_switch_big(PN_PR, 'o', 'O', 'i', 'I', PN1_O, PN2_I);
1216
1217
      /* pn-provider */
1218
76.4k
      param_switch(PN1_O, 'v', 'V', PN1_V);
1219
52.6k
      param_switch(PN1_V, 'i', 'I', PN1_I);
1220
45.7k
      param_switch(PN1_I, 'd', 'D', PN1_D);
1221
41.4k
      param_switch(PN1_D, 'e', 'E', PN1_E);
1222
35.4k
      param_switch(PN1_E, 'r', 'R', PN1_FIN);
1223
30.0k
      case PN1_FIN:
1224
26.7k
        param=&uri->pn_provider;
1225
26.7k
        switch(*p){
1226
82
          case '@':
1227
82
            still_at_user;
1228
43
            break;
1229
23.5k
          case '=':
1230
23.5k
            state=PN1_eq;
1231
23.5k
            break;
1232
1.13k
          semicolon_case;
1233
1.13k
            uri->pn_provider.s=b;
1234
1.13k
            uri->pn_provider.len=(p-b);
1235
1.13k
            break;
1236
95
          question_case;
1237
95
            uri->pn_provider.s=b;
1238
95
            uri->pn_provider.len=(p-b);
1239
95
            break;
1240
683
          colon_case;
1241
683
            break;
1242
1.11k
          default:
1243
1.11k
            state=URI_PARAM_P;
1244
26.7k
        }
1245
26.6k
        break;
1246
        /* handle pn-provider=something case */
1247
26.6k
      case PN1_eq:
1248
23.5k
        param=&uri->pn_provider;
1249
23.5k
        param_val=&uri->pn_provider_val;
1250
23.5k
        switch(*p){
1251
82
          param_common_cases;
1252
16.0k
          default:
1253
16.0k
            v=p;
1254
16.0k
            state=URI_VAL_P;
1255
23.5k
        }
1256
23.5k
        break;
1257
1258
      /* pn-prid */
1259
23.5k
      param_switch(PN2_I, 'd', 'D', PN2_D);
1260
14.9k
      param_xswitch1(PN2_D, '=', PN2_eq);
1261
1.99k
      case PN2_eq:
1262
1.95k
        param=&uri->pn_prid;
1263
1.95k
        param_val=&uri->pn_prid_val;
1264
1.95k
        switch(*p){
1265
61
          param_common_cases;
1266
355
          default:
1267
355
            v=p;
1268
355
            state=URI_VAL_P;
1269
1.95k
        }
1270
1.91k
        break;
1271
1272
      /* pn-param */
1273
20.8k
      param_switch(PN3_A, 'r', 'R', PN3_R);
1274
20.8k
      param_switch(PN3_R, 'a', 'A', PN3_A2);
1275
14.8k
      param_switch(PN3_A2, 'm', 'M', PN3_M);
1276
12.1k
      param_xswitch1(PN3_M, '=', PN3_eq);
1277
8.32k
      case PN3_eq:
1278
8.29k
        param=&uri->pn_param;
1279
8.29k
        param_val=&uri->pn_param_val;
1280
8.29k
        switch(*p){
1281
82
          param_common_cases;
1282
509
          default:
1283
509
            v=p;
1284
509
            state=URI_VAL_P;
1285
8.29k
        }
1286
8.26k
        break;
1287
1288
      /* pn-purr */
1289
21.4k
      param_switch(PN4_U, 'r', 'R', PN4_R);
1290
21.4k
      param_switch(PN4_R, 'r', 'R', PN4_R2);
1291
11.6k
      param_xswitch1(PN4_R2, '=', PN4_eq);
1292
5.00k
      case PN4_eq:
1293
4.96k
        param=&uri->pn_purr;
1294
4.96k
        param_val=&uri->pn_purr_val;
1295
4.96k
        switch(*p){
1296
64
          param_common_cases;
1297
567
          default:
1298
567
            v=p;
1299
567
            state=URI_VAL_P;
1300
4.96k
        }
1301
4.94k
        break;
1302
1303
1304
81.1k
      case URI_HEADERS:
1305
        /* for now nobody needs them so we completely ignore the
1306
         * headers (they are not allowed in request uri) --andrei */
1307
81.1k
        switch(*p){
1308
7.68k
          case '@':
1309
            /* yak, we are still at user */
1310
7.68k
            still_at_user;
1311
4.87k
            break;
1312
11.8k
          case ';':
1313
            /* we might be still parsing user, try it */
1314
11.8k
            if (found_user) goto error_bad_char;
1315
5.40k
            error_headers=1; /* if this is not the user
1316
                      we have an error */
1317
            /* if pass is set => it cannot be user:pass
1318
             * => error (';') is illegal in a header */
1319
5.40k
            if (pass) goto error_headers;
1320
5.18k
            break;
1321
5.18k
          case ':':
1322
4.06k
            if (found_user==0){
1323
              /*might be pass but only if user not found yet*/
1324
2.18k
              if (pass){
1325
94
                found_user=1; /* no user */
1326
94
                pass=0;
1327
2.08k
              }else{
1328
2.08k
                pass=p;
1329
2.08k
              }
1330
2.18k
            }
1331
4.06k
            break;
1332
3.46k
          case '?':
1333
3.46k
            if (pass){
1334
45
              found_user=1; /* no user, pass cannot contain '?'*/
1335
45
              pass=0;
1336
45
            }
1337
3.46k
            break;
1338
81.1k
        }
1339
71.7k
        break;
1340
71.7k
      default:
1341
0
        goto error_bug;
1342
9.89M
    }
1343
9.89M
  }
1344
1345
  /*end of uri */
1346
24.9k
  switch (state){
1347
55
    case URI_INIT: /* error empty uri */
1348
55
      goto error_too_short;
1349
121
    case URI_USER:
1350
      /* this is the host, it can't be the user */
1351
121
      if (found_user) goto error_bad_uri;
1352
121
      uri->host.s=s;
1353
121
      uri->host.len=p-s;
1354
121
      state=URI_HOST;
1355
121
      break;
1356
2.79k
    case URI_PASSWORD:
1357
      /* this is the port, it can't be the passwd */
1358
2.79k
      if (found_user) goto error_bad_port;
1359
2.79k
      if (port_no > USHRT_MAX) goto error_bad_port;
1360
181
      uri->port.s=s;
1361
181
      uri->port.len=p-s;
1362
181
      uri->port_no=port_no;
1363
181
      uri->host=uri->user;
1364
181
      uri->user.s=0;
1365
181
      uri->user.len=0;
1366
181
      break;
1367
545
    case URI_PASSWORD_ALPHA:
1368
      /* this is the port, it can't be the passwd */
1369
545
      goto error_bad_port;
1370
497
    case URI_HOST_P:
1371
523
    case URI_HOST6_END:
1372
523
      uri->host.s=s;
1373
523
      uri->host.len=p-s;
1374
523
      break;
1375
9.39k
    case URI_HOST: /* error: null host */
1376
9.51k
    case URI_HOST6_P: /* error: unterminated ipv6 reference*/
1377
9.51k
      goto error_bad_host;
1378
336
    case URI_PORT:
1379
336
      uri->port.s=s;
1380
336
      uri->port.len=p-s;
1381
336
      uri->port_no=port_no;
1382
336
      break;
1383
1.65k
    case URI_PARAM:
1384
4.02k
    case URI_PARAM_P:
1385
4.16k
    case URI_PARAM_VAL_P:
1386
4.16k
      u_param_set(b, v);
1387
    /* intermediate param states */
1388
4.20k
    case PT_T: /* transport */
1389
4.24k
    case PT_R:
1390
4.27k
    case PT_A:
1391
4.29k
    case PT_N:
1392
4.32k
    case PT_S:
1393
4.34k
    case PT_P:
1394
4.36k
    case PT_O:
1395
4.38k
    case PT_R2:
1396
4.40k
    case PT_T2:
1397
4.42k
    case PT_eq: /* ignore empty transport params */
1398
4.44k
    case PTTL_T2: /* ttl */
1399
4.46k
    case PTTL_L:
1400
4.47k
    case PTTL_eq:
1401
4.50k
    case PU_U:  /* user */
1402
4.52k
    case PU_S:
1403
4.55k
    case PU_E:
1404
4.57k
    case PU_R:
1405
4.58k
    case PU_eq:
1406
4.61k
    case PM_M: /* method */
1407
4.64k
    case PM_E:
1408
4.66k
    case PM_T:
1409
4.68k
    case PM_H:
1410
4.70k
    case PM_O:
1411
4.72k
    case PM_D:
1412
4.74k
    case PM_eq:
1413
4.76k
    case PLR_L: /* lr */
1414
4.80k
    case PR2_R:  /* r2 */
1415
4.83k
    case PG_G: /* gr */
1416
4.83k
      uri->params.s=s;
1417
4.83k
      uri->params.len=p-s;
1418
4.83k
      break;
1419
    /* fin param states */
1420
23
    case PLR_R_FIN:
1421
36
    case PLR_eq:
1422
36
      uri->params.s=s;
1423
36
      uri->params.len=p-s;
1424
36
      uri->lr.s=b;
1425
36
      uri->lr.len=p-b;
1426
36
      break;
1427
14
    case PR2_2_FIN:
1428
28
    case PR2_eq:
1429
28
      uri->params.s=s;
1430
28
      uri->params.len=p-s;
1431
28
      uri->r2.s=b;
1432
28
      uri->r2.len=p-b;
1433
28
      break;
1434
21
    case PG_G_FIN:
1435
32
    case PG_eq:
1436
32
      uri->params.s=s;
1437
32
      uri->params.len=p-s;
1438
32
      uri->gr.s=b;
1439
32
      uri->gr.len=p-b;
1440
32
      break;
1441
22
    case PN1_FIN:
1442
27
    case PN1_eq:
1443
27
      uri->params.s=s;
1444
27
      uri->params.len=p-s;
1445
27
      uri->pn_provider.s=b;
1446
27
      uri->pn_provider.len=p-b;
1447
27
      break;
1448
578
    case URI_VAL_P:
1449
    /* intermediate value states */
1450
600
    case VU_U:
1451
620
    case VU_D:
1452
635
    case VT_T:
1453
657
    case VT_C:
1454
671
    case VTLS_L:
1455
690
    case VS_S:
1456
707
    case VS_C:
1457
725
    case VW_W:
1458
741
    case VS_T:
1459
741
      uri->params.s=s;
1460
741
      uri->params.len=p-s;
1461
741
      param_set(b, v);
1462
741
      break;
1463
    /* fin value states */
1464
16
    case VU_P_FIN:
1465
16
      uri->params.s=s;
1466
16
      uri->params.len=p-s;
1467
16
      param_set(b, v);
1468
16
      uri->proto=PROTO_UDP;
1469
16
      break;
1470
17
    case VT_P_FIN:
1471
17
      uri->params.s=s;
1472
17
      uri->params.len=p-s;
1473
17
      param_set(b, v);
1474
17
      uri->proto=PROTO_TCP;
1475
17
      break;
1476
14
    case VTLS_S_FIN:
1477
14
      uri->params.s=s;
1478
14
      uri->params.len=p-s;
1479
14
      param_set(b, v);
1480
14
      uri->proto=PROTO_TLS;
1481
14
      break;
1482
16
    case VS_P_FIN:
1483
16
      uri->params.s=s;
1484
16
      uri->params.len=p-s;
1485
16
      param_set(b, v);
1486
16
      uri->proto=PROTO_SCTP;
1487
16
      break;
1488
17
    case VW_S:
1489
17
    case VW_S_FIN:
1490
17
      uri->params.s=s;
1491
17
      uri->params.len=p-s;
1492
17
      param_set(b, v);
1493
17
      uri->proto=PROTO_WS;
1494
17
      break;
1495
24
    case VWS_S_FIN:
1496
24
      uri->params.s=s;
1497
24
      uri->params.len=p-s;
1498
24
      param_set(b, v);
1499
24
      uri->proto=PROTO_WSS;
1500
24
      break;
1501
    /* headers */
1502
4.23k
    case URI_HEADERS:
1503
4.23k
      uri->headers.s=s;
1504
4.23k
      uri->headers.len=p-s;
1505
4.23k
      if (error_headers) goto error_headers;
1506
1.94k
      break;
1507
    /* intermediate PN param states */
1508
1.94k
    case PN_P:
1509
73
    case PN_N:
1510
88
    case PN_dash:
1511
112
    case PN_P2:
1512
132
    case PN_PR:
1513
151
    case PN1_O:
1514
169
    case PN1_V:
1515
192
    case PN1_I:
1516
212
    case PN1_D:
1517
226
    case PN1_E:
1518
247
    case PN2_I:
1519
266
    case PN3_A:
1520
286
    case PN3_R:
1521
303
    case PN3_A2:
1522
324
    case PN4_U:
1523
339
    case PN4_R:
1524
339
      uri->params.s=s;
1525
339
      uri->params.len=p-s;
1526
339
      break;
1527
105
    case PN2_D:
1528
144
    case PN2_eq:
1529
217
    case PN3_M:
1530
242
    case PN3_eq:
1531
301
    case PN4_R2:
1532
345
    case PN4_eq:
1533
345
      goto error_bad_uri;
1534
308
    default:
1535
308
      goto error_bug;
1536
24.9k
  }
1537
9.24k
  switch(uri->type){
1538
3.46k
    case TEL_URI_T:
1539
3.46k
    case TELS_URI_T:
1540
      /* fix tel uris, move the number in uri and empty the host */
1541
3.46k
      uri->user=uri->host;
1542
      /* TEL does not have a host part, still most of the code expects
1543
       * one, so lets keep the pointer, but set a 0 length */
1544
3.46k
      uri->host.len=0;
1545
3.46k
      break;
1546
5.71k
    case SIP_URI_T:
1547
5.73k
    case SIPS_URI_T:
1548
5.74k
    case URN_SERVICE_URI_T:
1549
      /* nothing to do for these URIs */
1550
5.74k
      break;
1551
38
    case URN_NENA_SERVICE_URI_T:
1552
38
      uri->user.s=0;
1553
38
      uri->user.len=0;
1554
      /* keep the service name as host part */
1555
38
      break;
1556
0
    case ERROR_URI_T:
1557
0
      LM_ERR("unexpected error (BUG?)\n");
1558
0
      goto error_bad_uri;
1559
0
      break; /* do nothing, avoids a compilation warning */
1560
9.24k
  }
1561
#ifdef EXTRA_DEBUG
1562
  /* do stuff */
1563
  LM_DBG("parsed uri:\n type=%d user=<%.*s>(%d)\n passwd=<%.*s>(%d)\n"
1564
      " host=<%.*s>(%d)\n port=<%.*s>(%d): %d\n params=<%.*s>(%d)\n"
1565
      " headers=<%.*s>(%d)\n",
1566
      uri->type,
1567
      uri->user.len, ZSW(uri->user.s), uri->user.len,
1568
      uri->passwd.len, ZSW(uri->passwd.s), uri->passwd.len,
1569
      uri->host.len, ZSW(uri->host.s), uri->host.len,
1570
      uri->port.len, ZSW(uri->port.s), uri->port.len, uri->port_no,
1571
      uri->params.len, ZSW(uri->params.s), uri->params.len,
1572
      uri->headers.len, ZSW(uri->headers.s), uri->headers.len
1573
    );
1574
  LM_DBG(" uri params:\n   transport=<%.*s>, val=<%.*s>, proto=%d\n",
1575
      uri->transport.len, ZSW(uri->transport.s), uri->transport_val.len,
1576
      ZSW(uri->transport_val.s), uri->proto);
1577
  LM_DBG("   user-param=<%.*s>, val=<%.*s>\n",
1578
      uri->user_param.len, ZSW(uri->user_param.s),
1579
      uri->user_param_val.len, ZSW(uri->user_param_val.s));
1580
  LM_DBG("   method=<%.*s>, val=<%.*s>\n",
1581
      uri->method.len, ZSW(uri->method.s),
1582
      uri->method_val.len, ZSW(uri->method_val.s));
1583
  LM_DBG("   ttl=<%.*s>, val=<%.*s>\n",
1584
      uri->ttl.len, ZSW(uri->ttl.s),
1585
      uri->ttl_val.len, ZSW(uri->ttl_val.s));
1586
  LM_DBG("   maddr=<%.*s>, val=<%.*s>\n",
1587
      uri->maddr.len, ZSW(uri->maddr.s),
1588
      uri->maddr_val.len, ZSW(uri->maddr_val.s));
1589
  LM_DBG("   lr=<%.*s>, val=<%.*s>\n", uri->lr.len, ZSW(uri->lr.s),
1590
      uri->lr_val.len, ZSW(uri->lr_val.s));
1591
  LM_DBG("   r2=<%.*s>, val=<%.*s>\n", uri->r2.len, ZSW(uri->r2.s),
1592
      uri->r2_val.len, ZSW(uri->r2_val.s));
1593
  for(i=0; i<URI_MAX_U_PARAMS && uri->u_name[i].s; i++)
1594
    LM_DBG("uname=[%p]-><%.*s> uval=[%p]-><%.*s>\n",
1595
      uri->u_name[i].s, uri->u_name[i].len, uri->u_name[i].s,
1596
      uri->u_val[i].s, uri->u_val[i].len, uri->u_val[i].s);
1597
  if (i!=uri->u_params_no)
1598
    LM_ERR("inconsisten # of u_name:[%d]!=[%d]\n", i, uri->u_params_no);
1599
#endif
1600
9.24k
  return 0;
1601
1602
65.4k
error_too_short:
1603
65.4k
  LM_ERR("uri too short: <%.*s> (%d)\n",
1604
65.4k
      len, ZSW(buf), len);
1605
65.4k
  goto error_exit;
1606
25.3k
error_bad_char:
1607
25.3k
  LM_ERR("bad char '%c' in state %d"
1608
25.3k
      " parsed: <%.*s> (%d) / <%.*s> (%d)\n",
1609
25.3k
      p < end ? *p : *(buf+len-1), state, (int)(p-buf), ZSW(buf),
1610
0
      (int)(p-buf), len, ZSW(buf), len);
1611
25.3k
  goto error_exit;
1612
10.6k
error_bad_host:
1613
10.6k
  LM_ERR("bad host in uri (error at char %c in"
1614
10.6k
      " state %d) parsed: <%.*s>(%d) /<%.*s> (%d)\n",
1615
10.6k
      p < end ? *p : *(buf+len-1), state, (int)(p-buf), ZSW(buf),
1616
0
      (int)(p-buf), len, ZSW(buf), len);
1617
10.6k
  goto error_exit;
1618
5.27k
error_bad_port:
1619
5.27k
  LM_ERR("bad port in uri (error at char '%c' in"
1620
5.27k
      " state %d) parsed: <%.*s>(%d) /<%.*s> (%d)\n",
1621
5.27k
      p < end ? *p : *(buf+len-1), state, (int)(p-buf), ZSW(buf),
1622
0
      (int)(p-buf), len, ZSW(buf), len);
1623
5.27k
  goto error_exit;
1624
5.99k
error_bad_uri:
1625
5.99k
  LM_ERR("bad uri, state %d parsed: <%.*s> (%d) / <%.*s> (%d)\n",
1626
5.99k
       state, (int)(p-buf), ZSW(buf), (int)(p-buf), len,
1627
0
       ZSW(buf), len);
1628
5.99k
  goto error_exit;
1629
2.52k
error_headers:
1630
2.52k
  LM_ERR("bad uri headers: <%.*s>(%d) / <%.*s>(%d)\n",
1631
2.52k
      uri->headers.len, ZSW(uri->headers.s), uri->headers.len,
1632
0
      len, ZSW(buf), len);
1633
2.52k
  goto error_exit;
1634
308
error_bug:
1635
308
  LM_ERR("bad state %d parsed: <%.*s> (%d) / <%.*s> (%d)\n",
1636
308
       state, (int)(p-buf), ZSW(buf), (int)(p-buf), len, ZSW(buf), len);
1637
115k
error_exit:
1638
115k
  ser_error=E_BAD_URI;
1639
115k
  uri->type=ERROR_URI_T;
1640
115k
  update_stat(bad_URIs, 1);
1641
115k
  return E_BAD_URI;
1642
308
}
1643
1644
1645
int parse_sip_msg_uri(struct sip_msg* msg)
1646
63.4k
{
1647
63.4k
  char* tmp;
1648
63.4k
  int tmp_len;
1649
63.4k
  if (msg->parsed_uri_ok) return 1;
1650
1651
63.4k
  if (msg->new_uri.s){
1652
0
    tmp=msg->new_uri.s;
1653
0
    tmp_len=msg->new_uri.len;
1654
63.4k
  }else{
1655
63.4k
    tmp=msg->first_line.u.request.uri.s;
1656
63.4k
    tmp_len=msg->first_line.u.request.uri.len;
1657
63.4k
  }
1658
63.4k
  if (parse_uri(tmp, tmp_len, &msg->parsed_uri)<0){
1659
61.5k
    LM_ERR("bad uri <%.*s>\n", tmp_len, tmp);
1660
61.5k
    msg->parsed_uri_ok=0;
1661
61.5k
    set_err_info(OSER_EC_PARSER, OSER_EL_MEDIUM, "error parsing r-uri");
1662
61.5k
    set_err_reply(400, "bad r-uri");
1663
61.5k
    return -1;
1664
61.5k
  }
1665
1.91k
  msg->parsed_uri_ok=1;
1666
1.91k
  return 0;
1667
63.4k
}
1668
1669
1670
int parse_orig_ruri(struct sip_msg* msg)
1671
48.0k
{
1672
48.0k
  str *uri;
1673
1674
48.0k
  if (msg->parsed_orig_ruri_ok)
1675
0
    return 1;
1676
1677
48.0k
  uri = &REQ_LINE(msg).uri;
1678
1679
48.0k
  if (parse_uri(uri->s, uri->len, &msg->parsed_orig_ruri)<0) {
1680
46.1k
    LM_ERR("bad uri <%.*s>\n", uri->len, ZSW(uri->s));
1681
46.1k
    msg->parsed_orig_ruri_ok = 0;
1682
46.1k
    set_err_info(OSER_EC_PARSER, OSER_EL_MEDIUM,
1683
46.1k
        "error parsing incoming uri");
1684
46.1k
    set_err_reply(400, "bad i-uri");
1685
46.1k
    return -1;
1686
46.1k
  }
1687
1688
1.91k
  msg->parsed_orig_ruri_ok = 1;
1689
1.91k
  return 0;
1690
48.0k
}
1691
1692
#define compare_uri_val(field,cmpfunc) \
1693
0
  do { \
1694
0
    if (first.field.len != second.field.len) \
1695
0
    { \
1696
0
      LM_DBG("Different URI field - " #field "\n"); \
1697
0
      return 1; \
1698
0
    } \
1699
0
    else \
1700
0
    { \
1701
0
      if (first.field.len != 0) \
1702
0
        if (cmpfunc(first.field.s,second.field.s,first.field.len)) \
1703
0
        { \
1704
0
          LM_DBG("Different URI field - " #field "\n"); \
1705
0
          return 1; \
1706
0
        } \
1707
0
    } \
1708
0
  } while (0)
1709
1710
/* Compare 2 SIP URIs according to RFC 3261
1711
 *
1712
 * Return value : 0 if URIs match
1713
 *          1 if URIs don't match
1714
 *         -1 if errors have occurred
1715
 */
1716
int compare_uris(str *raw_uri_a,struct sip_uri* parsed_uri_a,
1717
          str *raw_uri_b,struct sip_uri *parsed_uri_b)
1718
0
{
1719
0
  #define UNESCAPED_BUF_LEN 1024
1720
0
  char unescaped_a[UNESCAPED_BUF_LEN], unescaped_b[UNESCAPED_BUF_LEN];
1721
1722
0
  str unescaped_userA={unescaped_a, UNESCAPED_BUF_LEN};
1723
0
  str unescaped_userB={unescaped_b, UNESCAPED_BUF_LEN};
1724
1725
0
  struct sip_uri first;
1726
0
  struct sip_uri second;
1727
0
  char matched[URI_MAX_U_PARAMS];
1728
0
  int i,j;
1729
1730
0
  if ( (!raw_uri_a && !parsed_uri_a) || (!raw_uri_b && !parsed_uri_b) )
1731
0
  {
1732
0
    LM_ERR("Provide either a raw or parsed form of a SIP URI\n");
1733
0
    return -1;
1734
0
  }
1735
1736
0
  if (raw_uri_a && raw_uri_b)
1737
0
  {
1738
1739
    /* maybe we're lucky and straight-forward comparison succeeds */
1740
0
    if (raw_uri_a->len == raw_uri_b->len)
1741
0
      if (strncasecmp(raw_uri_a->s,raw_uri_b->s,raw_uri_a->len) == 0)
1742
0
      {
1743
0
        LM_DBG("straight-forward URI match\n");
1744
0
        if (parse_uri(raw_uri_a->s,raw_uri_a->len,&first) < 0)
1745
0
        {
1746
0
          LM_ERR("Failed to parse first URI\n");
1747
0
          return -1;
1748
0
        }
1749
0
        if (parse_uri(raw_uri_b->s,raw_uri_b->len,&second) < 0)
1750
0
        {
1751
0
          LM_ERR("Failed to parse second URI\n");
1752
0
          return -1;
1753
0
        }
1754
0
        if (unescape_user(&first.user, &unescaped_userA) < 0 ||
1755
0
            unescape_user(&second.user, &unescaped_userB) < 0) {
1756
0
          LM_ERR("Failed to unescape user!\n");
1757
0
          return -1;
1758
0
        }
1759
0
        first.user = unescaped_userA;
1760
0
        second.user = unescaped_userB;
1761
0
        compare_uri_val(user,strncmp);
1762
0
        compare_uri_val(passwd,strncmp);
1763
0
        return 0;
1764
0
      }
1765
0
  }
1766
1767
  /* XXX - maybe if we have two parsed sip_uris,
1768
   * or only one parsed and one raw,
1769
   * it should be possible to do a straight-forward
1770
   * URI match ? */
1771
1772
0
  if (parsed_uri_a)
1773
0
    first = *parsed_uri_a;
1774
0
  else
1775
0
  {
1776
0
    if (parse_uri(raw_uri_a->s,raw_uri_a->len,&first) < 0)
1777
0
    {
1778
0
      LM_ERR("Failed to parse first URI\n");
1779
0
      return -1;
1780
0
    }
1781
0
  }
1782
1783
0
  if (parsed_uri_b)
1784
0
    second = *parsed_uri_b;
1785
0
  else
1786
0
  {
1787
0
    if (parse_uri(raw_uri_b->s,raw_uri_b->len,&second) < 0)
1788
0
    {
1789
0
      LM_ERR("Failed to parse second URI\n");
1790
0
      return -1;
1791
0
    }
1792
0
  }
1793
1794
0
  if (first.type != second.type)
1795
0
  {
1796
0
    LM_DBG("Different uri types\n");
1797
0
    return 1;
1798
0
  }
1799
1800
0
  if (unescape_user(&first.user, &unescaped_userA) < 0 ||
1801
0
      unescape_user(&second.user, &unescaped_userB) < 0) {
1802
0
    LM_ERR("Failed to unescape user!\n");
1803
0
    return -1;
1804
0
  }
1805
1806
0
  first.user = unescaped_userA;
1807
0
  second.user = unescaped_userB;
1808
1809
0
  compare_uri_val(user,strncmp);
1810
0
  compare_uri_val(passwd,strncmp);
1811
0
  compare_uri_val(host,strncasecmp);
1812
0
  compare_uri_val(port,strncmp);
1813
1814
0
  compare_uri_val(transport_val,strncasecmp);
1815
0
  compare_uri_val(ttl_val,strncasecmp);
1816
0
  compare_uri_val(user_param_val,strncasecmp);
1817
0
  compare_uri_val(maddr_val,strncasecmp);
1818
0
  compare_uri_val(method_val,strncasecmp);
1819
0
  compare_uri_val(lr_val,strncasecmp);
1820
0
  compare_uri_val(r2_val,strncasecmp);
1821
0
  compare_uri_val(gr_val,strncasecmp);
1822
0
  compare_uri_val(pn_provider_val,strncasecmp);
1823
0
  compare_uri_val(pn_prid_val,strncasecmp);
1824
0
  compare_uri_val(pn_param_val,strncasecmp);
1825
0
  compare_uri_val(pn_purr_val,strncasecmp);
1826
1827
0
  if (first.u_params_no == 0 || second.u_params_no == 0)
1828
    /* one URI doesn't have other params,
1829
     * automatically all unknown params in other URI match
1830
     */
1831
0
    goto headers_check;
1832
1833
0
  memset(matched,0,URI_MAX_U_PARAMS);
1834
1835
0
  for (i=0;i<first.u_params_no;i++)
1836
0
    for (j=0;j<second.u_params_no;j++)
1837
0
      if (matched[j] == 0 &&
1838
0
        (first.u_name[i].len == second.u_name[j].len &&
1839
0
                strncasecmp(first.u_name[i].s,second.u_name[j].s,
1840
0
              first.u_name[i].len) == 0))
1841
0
        {
1842
                    /* point of no return - matching unknown parameter values */
1843
0
          if (first.u_val[i].len != second.u_val[j].len)
1844
0
          {
1845
0
            LM_DBG("Different URI param value for param %.*s\n",
1846
0
                first.u_name[i].len,first.u_name[i].s);
1847
0
            return 1;
1848
0
          }
1849
0
          else
1850
0
          {
1851
0
            if (first.u_val[i].len == 0)
1852
0
            {
1853
              /* no value for unknown params - match */
1854
0
              matched[j] = 1;
1855
0
              break;
1856
0
            }
1857
1858
0
            if (strncasecmp(first.u_val[i].s,second.u_val[j].s,
1859
0
              second.u_val[j].len))
1860
0
            {
1861
0
              LM_DBG("Different URI param value for param %.*s\n",
1862
0
                first.u_name[i].len,first.u_name[i].s);
1863
0
              return 1;
1864
0
            }
1865
0
            else
1866
0
            {
1867
0
              matched[j] = 1;
1868
0
              break;
1869
0
            }
1870
0
          }
1871
0
        }
1872
1873
  /* got here, it means all unknown params in first URI have been resolved
1874
    => first URI matched second URI, and the other way around
1875
  */
1876
1877
0
headers_check:
1878
   /* XXX Do we really care ? */
1879
0
  compare_uri_val(headers,strncasecmp);
1880
0
  return 0;
1881
0
}