569 lines
14 KiB
C
569 lines
14 KiB
C
/*
|
|
* This file is part of the Sofia-SIP package
|
|
*
|
|
* Copyright (C) 2005 Nokia Corporation.
|
|
*
|
|
* Contact: Pekka Pessi <pekka.pessi@nokia.com>
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public License
|
|
* as published by the Free Software Foundation; either version 2.1 of
|
|
* the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful, but
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
|
|
* 02110-1301 USA
|
|
*
|
|
*/
|
|
|
|
/**@ingroup sip_parser
|
|
* @CFILE sip_parser.c
|
|
*
|
|
* SIP parser.
|
|
*
|
|
* @author Pekka Pessi <Pekka.Pessi@nokia.com>.
|
|
*
|
|
* @date Created: Thu Oct 5 14:01:24 2000 ppessi
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
/* Avoid casting sip_t to msg_pub_t and sip_header_t to msg_header_t */
|
|
#define MSG_PUB_T struct sip_s
|
|
#define MSG_HDR_T union sip_header_u
|
|
|
|
#include <sofia-sip/su_tagarg.h>
|
|
#include "sofia-sip/sip_parser.h"
|
|
#include <sofia-sip/msg_mclass.h>
|
|
|
|
#include <stddef.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <stdio.h>
|
|
#include <assert.h>
|
|
#include <limits.h>
|
|
|
|
#ifndef UINT32_MAX
|
|
#define UINT32_MAX (0xffffffffU)
|
|
#endif
|
|
|
|
/** Version of the SIP module */
|
|
char const sip_parser_version[] = VERSION;
|
|
|
|
/** SIP version 2.0. */
|
|
char const sip_version_2_0[] = "SIP/2.0";
|
|
|
|
/** Default message class */
|
|
extern msg_mclass_t const sip_mclass[];
|
|
|
|
msg_mclass_t const *sip_default_mclass(void)
|
|
{
|
|
return sip_mclass;
|
|
}
|
|
|
|
/** Extract the SIP message body, including separator line.
|
|
*
|
|
* @param msg message object [IN]
|
|
* @param sip public SIP message structure [IN/OUT]
|
|
* @param b buffer containing unparsed data [IN]
|
|
* @param bsiz buffer size [IN]
|
|
* @param eos true if buffer contains whole message [IN]
|
|
*
|
|
* @retval -1 error
|
|
* @retval 0 cannot proceed
|
|
* @retval m
|
|
*/
|
|
issize_t sip_extract_body(msg_t *msg, sip_t *sip, char b[], isize_t bsiz, int eos)
|
|
{
|
|
ssize_t m = 0;
|
|
size_t body_len;
|
|
|
|
if (!(sip->sip_flags & MSG_FLG_BODY)) {
|
|
/* We are looking at a potential empty line */
|
|
m = msg_extract_separator(msg, (msg_pub_t *)sip, b, bsiz, eos);
|
|
if (m <= 0)
|
|
return m;
|
|
sip->sip_flags |= MSG_FLG_BODY;
|
|
b += m;
|
|
bsiz -= m;
|
|
}
|
|
|
|
if (sip->sip_content_length)
|
|
body_len = sip->sip_content_length->l_length;
|
|
else if (MSG_IS_MAILBOX(sip->sip_flags)) /* message fragments */
|
|
body_len = 0;
|
|
else if (eos)
|
|
body_len = bsiz;
|
|
else if (bsiz == 0)
|
|
return m;
|
|
else
|
|
return -1;
|
|
|
|
if (body_len == 0) {
|
|
sip->sip_flags |= MSG_FLG_COMPLETE;
|
|
return m;
|
|
}
|
|
|
|
if (m)
|
|
return m;
|
|
|
|
if (eos && body_len > bsiz) {
|
|
sip->sip_flags |= MSG_FLG_TRUNC | MSG_FLG_ERROR;
|
|
return bsiz;
|
|
}
|
|
|
|
if ((m = msg_extract_payload(msg, (msg_pub_t *)sip,
|
|
NULL, body_len, b, bsiz, eos)) == -1)
|
|
return -1;
|
|
|
|
sip->sip_flags |= MSG_FLG_FRAGS;
|
|
if (bsiz >= body_len)
|
|
sip->sip_flags |= MSG_FLG_COMPLETE;
|
|
|
|
return m;
|
|
}
|
|
|
|
/** Parse SIP version.
|
|
*
|
|
* Parse a SIP version string. Update the
|
|
* pointer at @a ss to first non-LWS character after the version string.
|
|
*
|
|
* @param ss string to be parsed [IN/OUT]
|
|
* @param ver value result for version [OUT]
|
|
*
|
|
* @retval 0 when successful,
|
|
* @retval -1 upon an error.
|
|
*/
|
|
int sip_version_d(char **ss, char const **ver)
|
|
{
|
|
char *s = *ss;
|
|
char const *result;
|
|
size_t const version_size = sizeof(sip_version_2_0) - 1;
|
|
|
|
if (strncasecmp(s, sip_version_2_0, version_size) == 0 &&
|
|
!IS_TOKEN(s[version_size])) {
|
|
result = sip_version_2_0;
|
|
s += version_size;
|
|
}
|
|
else {
|
|
/* Version consists of two tokens, separated by / */
|
|
size_t l1 = 0, l2 = 0, n;
|
|
|
|
result = s;
|
|
|
|
l1 = span_token(s);
|
|
for (n = l1; IS_LWS(s[n]); n++);
|
|
if (s[n] == '/') {
|
|
for (n++; IS_LWS(s[n]); n++);
|
|
l2 = span_token(s + n);
|
|
n += l2;
|
|
}
|
|
|
|
if (l1 == 0 || l2 == 0)
|
|
return -1;
|
|
|
|
/* If there is extra ws between tokens, compact version */
|
|
if (n > l1 + 1 + l2) {
|
|
s[l1] = '/';
|
|
memmove(s + l1 + 1, s + n - l2, l2);
|
|
s[l1 + 1 + l2] = 0;
|
|
|
|
/* Compare again with compacted version */
|
|
if (strcasecmp(s, sip_version_2_0) == 0)
|
|
result = sip_version_2_0;
|
|
}
|
|
|
|
s += n;
|
|
}
|
|
|
|
while (IS_WS(*s)) *s++ = '\0';
|
|
|
|
*ss = s;
|
|
|
|
if (ver)
|
|
*ver = result;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/** Calculate extra space required by version string */
|
|
isize_t sip_version_xtra(char const *version)
|
|
{
|
|
if (version == SIP_VERSION_CURRENT)
|
|
return 0;
|
|
return MSG_STRING_SIZE(version);
|
|
}
|
|
|
|
/** Duplicate a transport string */
|
|
void sip_version_dup(char **pp, char const **dd, char const *s)
|
|
{
|
|
if (s == SIP_VERSION_CURRENT)
|
|
*dd = s;
|
|
else
|
|
MSG_STRING_DUP(*pp, *dd, s);
|
|
}
|
|
|
|
char const sip_method_name_invite[] = "INVITE";
|
|
char const sip_method_name_ack[] = "ACK";
|
|
char const sip_method_name_cancel[] = "CANCEL";
|
|
char const sip_method_name_bye[] = "BYE";
|
|
char const sip_method_name_options[] = "OPTIONS";
|
|
char const sip_method_name_register[] = "REGISTER";
|
|
char const sip_method_name_info[] = "INFO";
|
|
char const sip_method_name_prack[] = "PRACK";
|
|
char const sip_method_name_update[] = "UPDATE";
|
|
char const sip_method_name_message[] = "MESSAGE";
|
|
char const sip_method_name_subscribe[] = "SUBSCRIBE";
|
|
char const sip_method_name_notify[] = "NOTIFY";
|
|
char const sip_method_name_refer[] = "REFER";
|
|
char const sip_method_name_publish[] = "PUBLISH";
|
|
|
|
/** Well-known SIP method names. */
|
|
char const * const sip_method_names[] = {
|
|
"<UNKNOWN>",
|
|
sip_method_name_invite,
|
|
sip_method_name_ack,
|
|
sip_method_name_cancel,
|
|
sip_method_name_bye,
|
|
sip_method_name_options,
|
|
sip_method_name_register,
|
|
sip_method_name_info,
|
|
sip_method_name_prack,
|
|
sip_method_name_update,
|
|
sip_method_name_message,
|
|
sip_method_name_subscribe,
|
|
sip_method_name_notify,
|
|
sip_method_name_refer,
|
|
sip_method_name_publish,
|
|
/* If you add something here, add also them to sip_method_d! */
|
|
NULL
|
|
};
|
|
|
|
/** Get canonic method name. */
|
|
char const *sip_method_name(sip_method_t method, char const *name)
|
|
{
|
|
const size_t N = sizeof(sip_method_names)/sizeof(sip_method_names[0]);
|
|
if (method > 0 && (size_t)method <= N)
|
|
return sip_method_names[method];
|
|
else if (method == 0)
|
|
return name;
|
|
else
|
|
return NULL;
|
|
}
|
|
|
|
/**Parse a SIP method name.
|
|
*
|
|
* Parse a SIP method name and return a code corresponding to the method.
|
|
* The address of the first non-LWS character after method name is stored in
|
|
* @a *ss.
|
|
*
|
|
* @param ss pointer to pointer to string to be parsed
|
|
* @param return_name value-result parameter for method name
|
|
*
|
|
* @note
|
|
* If there is no whitespace after method name, the value in @a *return_name
|
|
* may not be NUL-terminated. The calling function @b must NUL terminate
|
|
* the value by setting the @a **ss to NUL after first examining its value.
|
|
*
|
|
* @return The method code if method
|
|
* was identified, 0 (sip_method_unknown()) if method is not known, or @c -1
|
|
* (sip_method_invalid()) if an error occurred.
|
|
*
|
|
* If the value-result argument @a return_name is not @c NULL,
|
|
* a pointer to the method name is stored to it.
|
|
*/
|
|
sip_method_t sip_method_d(char **ss, char const **return_name)
|
|
{
|
|
char *s = *ss, c = *s;
|
|
char const *name;
|
|
int code = sip_method_unknown;
|
|
size_t n = 0;
|
|
|
|
#define MATCH(s, m) (strncmp(s, m, n = sizeof(m) - 1) == 0)
|
|
|
|
switch (c) {
|
|
case 'A': if (MATCH(s, "ACK")) code = sip_method_ack; break;
|
|
case 'B': if (MATCH(s, "BYE")) code = sip_method_bye; break;
|
|
case 'C':
|
|
if (MATCH(s, "CANCEL"))
|
|
code = sip_method_cancel;
|
|
break;
|
|
case 'I':
|
|
if (MATCH(s, "INVITE"))
|
|
code = sip_method_invite;
|
|
else if (MATCH(s, "INFO"))
|
|
code = sip_method_info;
|
|
break;
|
|
case 'M': if (MATCH(s, "MESSAGE")) code = sip_method_message; break;
|
|
case 'N': if (MATCH(s, "NOTIFY")) code = sip_method_notify; break;
|
|
case 'O': if (MATCH(s, "OPTIONS")) code = sip_method_options; break;
|
|
case 'P':
|
|
if (MATCH(s, "PRACK")) code = sip_method_prack;
|
|
else if (MATCH(s, "PUBLISH")) code = sip_method_publish;
|
|
break;
|
|
case 'R':
|
|
if (MATCH(s, "REGISTER"))
|
|
code = sip_method_register;
|
|
else if (MATCH(s, "REFER"))
|
|
code = sip_method_refer;
|
|
break;
|
|
case 'S':
|
|
if (MATCH(s, "SUBSCRIBE"))
|
|
code = sip_method_subscribe;
|
|
break;
|
|
case 'U':
|
|
if (MATCH(s, "UPDATE"))
|
|
code = sip_method_update;
|
|
break;
|
|
}
|
|
|
|
#undef MATCH
|
|
|
|
if (IS_NON_WS(s[n]))
|
|
/* Unknown method */
|
|
code = sip_method_unknown;
|
|
|
|
if (code == sip_method_unknown) {
|
|
name = s;
|
|
for (n = 0; IS_UNRESERVED(s[n]); n++)
|
|
;
|
|
if (s[n]) {
|
|
if (!IS_LWS(s[n]))
|
|
return sip_method_invalid;
|
|
if (return_name)
|
|
s[n++] = '\0';
|
|
}
|
|
}
|
|
else {
|
|
name = sip_method_names[code];
|
|
}
|
|
|
|
while (IS_LWS(s[n]))
|
|
n++;
|
|
|
|
*ss = (s + n);
|
|
if (return_name) *return_name = name;
|
|
|
|
return code;
|
|
}
|
|
|
|
/** Get method enum corresponding to method name */
|
|
sip_method_t sip_method_code(char const *name)
|
|
{
|
|
/* Note that sip_method_d() does not change string if return_name is NULL */
|
|
return sip_method_d((char **)&name, NULL);
|
|
}
|
|
|
|
char const sip_transport_udp[] = "SIP/2.0/UDP";
|
|
char const sip_transport_tcp[] = "SIP/2.0/TCP";
|
|
char const sip_transport_sctp[] = "SIP/2.0/SCTP";
|
|
char const sip_transport_tls[] = "SIP/2.0/TLS";
|
|
|
|
/** Decode transport */
|
|
issize_t sip_transport_d(char **ss, char const **ttransport)
|
|
{
|
|
char const *transport;
|
|
char *pn, *pv, *pt;
|
|
size_t pn_len, pv_len, pt_len;
|
|
char *s = *ss;
|
|
|
|
#define TRANSPORT_MATCH(t) \
|
|
(strncasecmp(s+7, t+7, sizeof(t)-8) == 0 && (IS_LWS(s[sizeof(t)])) \
|
|
&& (transport = t, s += sizeof(t) - 1))
|
|
|
|
if (strncasecmp(s, "SIP/2.0", 7) != 0 ||
|
|
(!TRANSPORT_MATCH(sip_transport_udp) &&
|
|
!TRANSPORT_MATCH(sip_transport_tcp) &&
|
|
!TRANSPORT_MATCH(sip_transport_sctp) &&
|
|
!TRANSPORT_MATCH(sip_transport_tls))) {
|
|
/* Protocol name */
|
|
transport = pn = s;
|
|
skip_token(&s);
|
|
pn_len = s - pn;
|
|
skip_lws(&s);
|
|
if (pn_len == 0 || *s++ != '/') return -1;
|
|
skip_lws(&s);
|
|
|
|
/* Protocol version */
|
|
pv = s;
|
|
skip_token(&s);
|
|
pv_len = s - pv;
|
|
skip_lws(&s);
|
|
if (pv_len == 0 || *s++ != '/') return -1;
|
|
skip_lws(&s);
|
|
|
|
/* Transport protocol */
|
|
pt = s;
|
|
skip_token(&s);
|
|
pt_len = s - pt;
|
|
if (pt_len == 0) return -1;
|
|
|
|
/* Remove whitespace between protocol name and version */
|
|
if (pn + pn_len + 1 != pv) {
|
|
pn[pn_len] = '/';
|
|
pv = memmove(pn + pn_len + 1, pv, pv_len);
|
|
}
|
|
|
|
/* Remove whitespace between protocol version and transport */
|
|
if (pv + pv_len + 1 != pt) {
|
|
pv[pv_len] = '/';
|
|
pt = memmove(pv + pv_len + 1, pt, pt_len);
|
|
pt[pt_len] = '\0';
|
|
|
|
/* extra whitespace? */
|
|
if (!strcasecmp(transport, sip_transport_udp))
|
|
transport = sip_transport_udp;
|
|
else if (!strcasecmp(transport, sip_transport_tcp))
|
|
transport = sip_transport_tcp;
|
|
else if (!strcasecmp(transport, sip_transport_sctp))
|
|
transport = sip_transport_sctp;
|
|
else if (!strcasecmp(transport, sip_transport_tls))
|
|
transport = sip_transport_tls;
|
|
}
|
|
}
|
|
|
|
if (IS_LWS(*s)) { *s++ = '\0'; skip_lws(&s); }
|
|
*ss = s;
|
|
*ttransport = transport;
|
|
return 0;
|
|
}
|
|
|
|
/** Calculate extra space required by sip_transport_dup() */
|
|
isize_t sip_transport_xtra(char const *transport)
|
|
{
|
|
if (transport == sip_transport_udp ||
|
|
transport == sip_transport_tcp ||
|
|
transport == sip_transport_sctp ||
|
|
transport == sip_transport_tls ||
|
|
strcasecmp(transport, sip_transport_udp) == 0 ||
|
|
strcasecmp(transport, sip_transport_tcp) == 0 ||
|
|
strcasecmp(transport, sip_transport_sctp) == 0 ||
|
|
strcasecmp(transport, sip_transport_tls) == 0)
|
|
return 0;
|
|
|
|
return MSG_STRING_SIZE(transport);
|
|
}
|
|
|
|
/** Duplicate a transport string */
|
|
void sip_transport_dup(char **pp, char const **dd, char const *s)
|
|
{
|
|
if (s == sip_transport_udp)
|
|
*dd = s;
|
|
else if (s == sip_transport_tcp)
|
|
*dd = s;
|
|
else if (s == sip_transport_sctp)
|
|
*dd = s;
|
|
else if (s == sip_transport_tls)
|
|
*dd = s;
|
|
else if (strcasecmp(s, sip_transport_udp) == 0)
|
|
*dd = sip_transport_udp;
|
|
else if (strcasecmp(s, sip_transport_tcp) == 0)
|
|
*dd = sip_transport_tcp;
|
|
else if (strcasecmp(s, sip_transport_sctp) == 0)
|
|
*dd = sip_transport_sctp;
|
|
else if (strcasecmp(s, sip_transport_tls) == 0)
|
|
*dd = sip_transport_tls;
|
|
else
|
|
MSG_STRING_DUP(*pp, *dd, s);
|
|
}
|
|
|
|
/** Parse SIP <word "@" word> construct used in @CallID. */
|
|
char *sip_word_at_word_d(char **ss)
|
|
{
|
|
char *rv = *ss, *s0 = *ss;
|
|
|
|
skip_word(ss);
|
|
if (s0 == *ss)
|
|
return NULL;
|
|
if (**ss == '@') {
|
|
(*ss)++;
|
|
s0 = *ss;
|
|
skip_word(ss);
|
|
if (s0 == *ss)
|
|
return NULL;
|
|
}
|
|
if (IS_LWS(**ss))
|
|
(*ss)++;
|
|
skip_lws(ss);
|
|
|
|
return rv;
|
|
}
|
|
|
|
/**Add message separator, then test if message is complete.
|
|
*
|
|
* Add sip_content_length and sip_separator if they are missing.
|
|
* The test that all necessary message components ( @From, @To,
|
|
* @CSeq, @CallID, @ContentLength and message separator are present.
|
|
*
|
|
* @retval 0 when successful
|
|
* @retval -1 upon an error: headers are missing and they could not be added
|
|
*/
|
|
int sip_complete_message(msg_t *msg)
|
|
{
|
|
sip_t *sip = sip_object(msg);
|
|
su_home_t *home = msg_home(msg);
|
|
size_t len = 0;
|
|
ssize_t mplen;
|
|
|
|
if (sip == NULL)
|
|
return -1;
|
|
|
|
if (!sip->sip_separator)
|
|
sip->sip_separator = sip_separator_create(msg_home(msg));
|
|
|
|
if (sip->sip_multipart) {
|
|
sip_content_type_t *c = sip->sip_content_type;
|
|
msg_multipart_t *mp = sip->sip_multipart;
|
|
sip_common_t *head;
|
|
|
|
if (!c || msg_multipart_complete(msg_home(msg), c, mp) < 0)
|
|
return -1;
|
|
|
|
if (sip->sip_payload)
|
|
head = sip->sip_payload->pl_common;
|
|
else
|
|
head = sip->sip_separator->sep_common;
|
|
|
|
if (!head || !msg_multipart_serialize(&head->h_succ, mp))
|
|
return -1;
|
|
|
|
mplen = msg_multipart_prepare(msg, mp, sip->sip_flags);
|
|
if (mplen == -1)
|
|
return -1;
|
|
len = (size_t)mplen;
|
|
}
|
|
|
|
if (sip->sip_payload)
|
|
len += sip->sip_payload->pl_len;
|
|
|
|
if (len > UINT32_MAX)
|
|
return -1;
|
|
|
|
if (!sip->sip_content_length) {
|
|
msg_header_insert(msg, (msg_pub_t *)sip, (msg_header_t*)
|
|
sip_content_length_create(home, (uint32_t)len));
|
|
}
|
|
else {
|
|
if (sip->sip_content_length->l_length != len) {
|
|
sip->sip_content_length->l_length = (uint32_t)len;
|
|
sip_fragment_clear(sip->sip_content_length->l_common);
|
|
}
|
|
}
|
|
|
|
if (!sip->sip_cseq ||
|
|
!sip->sip_call_id ||
|
|
!sip->sip_to ||
|
|
!sip->sip_from ||
|
|
!sip->sip_separator ||
|
|
!sip->sip_content_length)
|
|
return -1;
|
|
|
|
return 0;
|
|
}
|