| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545 | /* * Copyright (C) 2001-2003 FhG Fokus * * This file is part of ser, a free SIP server. * * ser is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version * * For a license to use the ser software under conditions * other than those described here, or to purchase support for this * software, please contact iptel.org by e-mail at the following addresses: *    [email protected] * * ser 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA * * History * ------- *  2003-01-28  removed scratchpad (jiri) *  2003-02-28  scratchpad compatibility abandoned (jiri) *  2003-03-06  enum_request_method changed to begin with 1; *               0 reserved for invalid values; (jiri) *  2003-03-31  removed sip_msg->repl_add_rm (andrei) *  2003-04-01  2 macros added: GET_NEXT_HOP and GET_RURI (janakj) *  2003-04-04  structure for parsed inbound uri added (jiri) *  2003-04-11  updated the  sip_uri structure (lots of fields added) (andrei) *  2003-04-12  added msg_flags to sip_msg (andrei) *  2003-11-02  added diversion header field to sip_msg (jh) *  2004-11-08  added force_send_socket (andrei) *  2005-02-25  uri types added (sip, sips & tel)  (andrei) *  2006-04-20  uri comp member (only if USE_COMP is defined) (andrei) *  2006-11-10  check_transaction_quadruple inlined (andrei) *  2007-01-26  added date, identity, identity_info header fields *		to sip_msg (gergo) *  2007-03-14  added SIP_MSG_START(msg) macro *//*! \file  * \brief Parser :: ??? * * \ingroup parser */#ifndef msg_parser_h#define msg_parser_h#include "../comp_defs.h"#include "../str.h"#include "../lump_struct.h"#include "../flags.h"#include "../ip_addr.h"#include "../md5utils.h"#include "../config.h"#include "parse_def.h"#include "parse_cseq.h"#include "parse_via.h"#include "parse_fline.h"#include "parse_retry_after.h"#include "hf.h"#include "../error.h"/*! \name convenience short-cut macros *//*@{ */#define REQ_LINE(_msg) ((_msg)->first_line.u.request)#define REQ_METHOD first_line.u.request.method_value#define REPLY_STATUS first_line.u.reply.statuscode#define REPLY_CLASS(_reply) ((_reply)->REPLY_STATUS/100)/*@} *//*! \brief start of "actual" sip msg (start of first line) */#define SIP_MSG_START(m)	((m)->first_line.u.request.method.s)/*! \brief number methods as power of two to allow bitmap matching */typedef enum request_method {	METHOD_UNDEF=0,           /*!< 0 - --- */	METHOD_INVITE=1,          /*!< 1 - 2^0 */	METHOD_CANCEL=2,          /*!< 2 - 2^1 */	METHOD_ACK=4,             /*!< 4 - 2^2 */	METHOD_BYE=8,             /*!< 8 - 2^3 */	METHOD_INFO=16,           /*!< 16 - 2^4 */	METHOD_REGISTER=32,       /*!< 32 - 2^5 */	METHOD_SUBSCRIBE=64,      /*!< 64 - 2^6 */	METHOD_NOTIFY=128,        /*!< 128 - 2^7 */	METHOD_MESSAGE=256,       /*!< 256 - 2^8 */	METHOD_OPTIONS=512,       /*!< 512 - 2^9 */	METHOD_PRACK=1024,        /*!< 1024 - 2^10 */	METHOD_UPDATE=2048,       /*!< 2048 - 2^11 */	METHOD_REFER=4096,        /*!< 4096 - 2^12 */	METHOD_PUBLISH=8192,      /*!< 8192 - 2^13 */	METHOD_OTHER=16384        /*!< 16384 - 2^14 */} request_method_t;#define FL_FORCE_RPORT  (1 << 0)  /*!< force rport */#define FL_FORCE_ACTIVE (1 << 1)  /*!< force active SDP */#define FL_SDP_IP_AFS   (1 << 2)  /*!< SDP IP rewritten */#define FL_SDP_PORT_AFS (1 << 3)  /*!< SDP port rewritten */#define FL_SHM_CLONE    (1 << 4)  /*!< msg cloned in SHM as a single chunk */#define FL_TIMEOUT      (1 << 5)  /*!< message belongs to an "expired" branch									 (for failure route use) */#define FL_REPLIED      (1 << 6)  /*!< message branch received at least one reply									 (for failure route use) */#define FL_HASH_INDEX   (1 << 7)  /*!< msg->hash_index contains a valid value (tm use)*/#define FL_MTU_TCP_FB   (1 << 8)#define FL_MTU_TLS_FB   (1 << 9)#define FL_MTU_SCTP_FB  (1 << 10)#define FL_ADD_LOCAL_RPORT  (1 << 11) /*!< add 'rport' to local via hdr */#define FL_SDP_BODY     (1 << 12)  /*!< msg has SDP in body */#define FL_USE_UAC_FROM      (1<<13)  /* take FROM hdr from UAC instead of UAS*/#define FL_USE_UAC_TO        (1<<14)  /* take TO hdr from UAC instead of UAS */#define FL_TM_RPL_MATCHED    (1<<15)  /* tm matched reply already */#define FL_RPL_SUSPENDED     (1<<16)  /* for async reply processing *//* WARNING: Value (1 << 28) is temporarily reserved for use in kamailio call_control * module (flag  FL_USE_CALL_CONTROL )! *//* WARNING: Value (1 << 29) is temporarily reserved for use in kamailio acc * module (flag FL_REQ_UPSTREAM)! *//* WARNING: Value (1 << 30) is temporarily reserved for use in kamailio * media proxy module (flag FL_USE_MEDIA_PROXY)! *//* WARNING: Value (1 << 31) is temporarily reserved for use in kamailio * nat_traversal module (flag FL_DO_KEEPALIVE)! */#define FL_MTU_FB_MASK  (FL_MTU_TCP_FB|FL_MTU_TLS_FB|FL_MTU_SCTP_FB)#define IFISMETHOD(methodname,firstchar)                                  \if (  (*tmp==(firstchar) || *tmp==((firstchar) | 32)) &&                  \        strncasecmp( tmp+1, #methodname +1, methodname##_LEN-1)==0 &&     \        *(tmp+methodname##_LEN)==' ') {                                   \                fl->type=SIP_REQUEST;                                     \                fl->u.request.method.len=methodname##_LEN;                \                fl->u.request.method_value=METHOD_##methodname;           \                tmp=buffer+methodname##_LEN;                              \}#define IS_HTTP(req)                                                \    ((req)->first_line.u.request.version.len >= HTTP_VERSION_LEN && \    !strncasecmp((req)->first_line.u.request.version.s,             \		HTTP_VERSION, HTTP_VERSION_LEN))#define IS_SIP(req)                                                \    ((req)->first_line.u.request.version.len >= SIP_VERSION_LEN && \    !strncasecmp((req)->first_line.u.request.version.s,             \		SIP_VERSION, SIP_VERSION_LEN))#define IS_HTTP_REPLY(rpl)                                                \    ((rpl)->first_line.u.reply.version.len >= HTTP_VERSION_LEN && \    !strncasecmp((rpl)->first_line.u.reply.version.s,             \		HTTP_VERSION, HTTP_VERSION_LEN))#define IS_SIP_REPLY(rpl)                                                \    ((rpl)->first_line.u.reply.version.len >= SIP_VERSION_LEN && \    !strncasecmp((rpl)->first_line.u.reply.version.s,             \		SIP_VERSION, SIP_VERSION_LEN))/*! \brief * Return a URI to which the message should be really sent (not what should * be in the Request URI. The following fields are tried in this order: * 1) dst_uri * 2) new_uri * 3) first_line.u.request.uri */#define GET_NEXT_HOP(m) \(((m)->dst_uri.s && (m)->dst_uri.len) ? (&(m)->dst_uri) : \(((m)->new_uri.s && (m)->new_uri.len) ? (&(m)->new_uri) : (&(m)->first_line.u.request.uri)))/*! \brief * Return the Reqeust URI of a message. * The following fields are tried in this order: * 1) new_uri * 2) first_line.u.request.uri */#define GET_RURI(m) \(((m)->new_uri.s && (m)->new_uri.len) ? (&(m)->new_uri) : (&(m)->first_line.u.request.uri))enum _uri_type{ERROR_URI_T=0, SIP_URI_T, SIPS_URI_T, TEL_URI_T, TELS_URI_T, URN_URI_T};typedef enum _uri_type uri_type;enum _uri_flags{	URI_USER_NORMALIZE=1,	URI_SIP_USER_PHONE=2}; /* bit fields */typedef enum _uri_flags uri_flags;/*! \brief The SIP uri object */struct sip_uri {	str user;     /*!< Username */	str passwd;   /*!< Password */	str host;     /*!< Host name */	str port;     /*!< Port number */	str params;   /*!< Parameters */	str sip_params; /*!< Parameters of the sip: URI.			  * (If a tel: URI is embedded in a sip: URI, then			  * params points to the parameters of the tel: URI,			  * and sip_params to the parameters of the sip: URI. 			  */	str headers;	unsigned short port_no;	unsigned short proto; /*!< from transport */	uri_type type; /*!< uri scheme */	uri_flags flags;	/*!< parameters */	str transport;	str ttl;	str user_param;	str maddr;	str method;	str lr;	str r2; /*!< ser specific rr parameter */	str gr;	str transport_val; /*!< transport value */	str ttl_val;	 /*!< TTL value */	str user_param_val; /*!< User= param value */	str maddr_val; /*!< Maddr= param value */	str method_val; /*!< Method value */	str lr_val; /*!< lr value placeholder for lr=on a.s.o*/	str r2_val;	str gr_val;#ifdef USE_COMP	unsigned short comp;#endif};typedef struct sip_uri sip_uri_t;struct msg_body;typedef void (*free_msg_body_f)(struct msg_body** ptr);typedef enum msg_body_type {	MSG_BODY_UNKNOWN = 0,	MSG_BODY_SDP} msg_body_type_t;/*! \brief This structure represents a generic SIP message body, regardless of the * body type. *  * Body parsers are supposed to cast this structure to some other * body-type specific structure, but the body type specific structure must * retain msg_body_type variable and a pointer to the free function as the  * first two variables within the structure. */typedef struct msg_body {	msg_body_type_t type;	free_msg_body_f free;} msg_body_t;/* pre-declaration, to include sys/time.h in .c */struct timeval;/* structure for cached decoded flow for outbound */typedef struct ocd_flow {		int decoded;		struct receive_info rcv;} ocd_flow_t;/* structure holding fields that don't have to be cloned in shm * - its content is memset'ed to in shm clone * - add to msg_ldata_reset() if a field uses dynamic memory */typedef struct msg_ldata {	ocd_flow_t flow;} msg_ldata_t;/*! \brief The SIP message */typedef struct sip_msg {	unsigned int id;               /*!< message id, unique/process*/	int pid;                       /*!< process id */	struct timeval tval;           /*!< time value associated to message */	snd_flags_t fwd_send_flags;    /*!< send flags for forwarding */	snd_flags_t rpl_send_flags;    /*!< send flags for replies */	struct msg_start first_line;   /*!< Message first line */	struct via_body* via1;         /*!< The first via */	struct via_body* via2;         /*!< The second via */	struct hdr_field* headers;     /*!< All the parsed headers*/	struct hdr_field* last_header; /*!< Pointer to the last parsed header*/	hdr_flags_t parsed_flag;    /*!< Already parsed header field types */	     /* Via, To, CSeq, Call-Id, From, end of header*/	     /* pointers to the first occurrences of these headers;		  * everything is also saved in 'headers'		  * (WARNING: do not deallocate them twice!)*/	struct hdr_field* h_via1;	struct hdr_field* h_via2;	struct hdr_field* callid;	struct hdr_field* to;	struct hdr_field* cseq;	struct hdr_field* from;	struct hdr_field* contact;	struct hdr_field* maxforwards;	struct hdr_field* route;	struct hdr_field* record_route;	struct hdr_field* content_type;	struct hdr_field* content_length;	struct hdr_field* authorization;	struct hdr_field* expires;	struct hdr_field* proxy_auth;	struct hdr_field* supported;	struct hdr_field* require;	struct hdr_field* proxy_require;	struct hdr_field* unsupported;	struct hdr_field* allow;	struct hdr_field* event;	struct hdr_field* accept;	struct hdr_field* accept_language;	struct hdr_field* organization;	struct hdr_field* priority;	struct hdr_field* subject;	struct hdr_field* user_agent;	struct hdr_field* server;	struct hdr_field* content_disposition;	struct hdr_field* diversion;	struct hdr_field* rpid;	struct hdr_field* refer_to;	struct hdr_field* session_expires;	struct hdr_field* min_se;	struct hdr_field* sipifmatch;	struct hdr_field* subscription_state;	struct hdr_field* date;	struct hdr_field* identity;	struct hdr_field* identity_info;	struct hdr_field* pai;	struct hdr_field* ppi;	struct hdr_field* path;	struct hdr_field* privacy;	struct msg_body* body;	char* eoh;        /*!< pointer to the end of header (if found) or null */	char* unparsed;   /*!< here we stopped parsing*/	struct receive_info rcv; /*!< source & dest ip, ports, proto a.s.o*/	char* buf;        /*!< scratch pad, holds a modified message,					   *  via, etc. point into it */	unsigned int len; /*!< message len (orig) */	     /* modifications */	str new_uri; /*!< changed first line uri, when you change this	                don't forget to set parsed_uri_ok to 0*/	str dst_uri; /*!< Destination URI, must be forwarded to this URI if len != 0 */	/* current uri */	int parsed_uri_ok; /*!< 1 if parsed_uri is valid, 0 if not, set if to 0	                      if you modify the uri (e.g change new_uri)*/	struct sip_uri parsed_uri; /*!< speed-up > keep here the parsed uri*/	int parsed_orig_ruri_ok; /*!< 1 if parsed_orig_uri is valid, 0 if not, set if to 0                              if you modify the uri (e.g change new_uri)*/	struct sip_uri parsed_orig_ruri; /*!< speed-up > keep here the parsed orig uri*/	struct lump* add_rm;       /*!< used for all the forwarded requests/replies */	struct lump* body_lumps;     /*!< Lumps that update Content-Length */	struct lump_rpl *reply_lump; /*!< only for localy generated replies !!!*/	/*! \brief str add_to_branch;	   whatever whoever want to append to Via branch comes here */	char add_to_branch_s[MAX_BRANCH_PARAM_LEN];	int add_to_branch_len;	unsigned int  hash_index; /*!< index to TM hash table; stored in core to avoid unnecessary calculations */	unsigned int msg_flags; /*!< internal flags used by core */	flag_t flags; /*!< config flags */	str set_global_address;	str set_global_port;	struct socket_info* force_send_socket; /*!< force sending on this socket */	str path_vec;	str instance;	unsigned int reg_id;	str ruid;	str location_ua;	/* structure with fields that are needed for local processing	 * - not cloned to shm, reset to 0 in the clone */	msg_ldata_t ldv;	/* IMPORTANT: when adding new fields in this structure (sip_msg_t),	 * be sure it is freed in free_sip_msg() and it is cloned or reset	 * to shm structure for transaction - see sip_msg_clone.c. In tm	 * module, take care of these fields for faked environemt used for	 * runing failure handlers - see modules/tm/t_reply.c */} sip_msg_t;/*! \brief pointer to a fakes message which was never received ;   (when this message is "relayed", it is generated out    of the original request)*/#define FAKED_REPLY     ((struct sip_msg *) -1)extern int via_cnt;/** global  request flags. *  msg->msg_flags should be OR'ed with it before * a flag value is checked, e.g.: * if ((msg->msg_flags|global_req_flags) & FL_XXX) ... */extern unsigned int global_req_flags;int parse_msg(char* const buf, const unsigned int len, struct sip_msg* const msg);int parse_headers(struct sip_msg* const msg, const hdr_flags_t flags, const int next);char* get_hdr_field(char* const buf, char* const end, struct hdr_field* const hdr);void free_sip_msg(struct sip_msg* const msg);/*! \brief make sure all HFs needed for transaction identification have been   parsed; return 0 if those HFs can't be found*/inline static int check_transaction_quadruple(struct sip_msg* const msg){	if ( parse_headers(msg, HDR_FROM_F|HDR_TO_F|HDR_CALLID_F|HDR_CSEQ_F,0)!=-1		&& msg->from && msg->to && msg->callid && msg->cseq ) {		return 1;	} else {		ser_error=E_BAD_TUPEL;		return 0;	}}/*! \brief returns a pointer to the begining of the msg's body */inline static char* get_body(struct sip_msg* const msg){	int offset;	unsigned int len;	if ( parse_headers(msg, HDR_EOH_F, 0)==-1 )		return 0;	if (msg->unparsed){		len=(unsigned int)(msg->unparsed-msg->buf);	}else return 0;	if ((len+2<=msg->len) && (strncmp(CRLF,msg->unparsed,CRLF_LEN)==0) )		offset = CRLF_LEN;	else if ( (len+1<=msg->len) &&				(*(msg->unparsed)=='\n' || *(msg->unparsed)=='\r' ) )		offset = 1;	else		return 0;	return msg->unparsed + offset;}/*! \brief If the new_uri is set, then reset it */void reset_new_uri(struct sip_msg* const msg);/*! \brief * Make a private copy of the string and assign it to dst_uri */int set_dst_uri(struct sip_msg* const msg, const str* const uri);/*! \brief If the dst_uri is set to an URI then reset it */void reset_dst_uri(struct sip_msg* const msg);hdr_field_t* get_hdr(const sip_msg_t* const msg, const enum _hdr_types_t ht);hdr_field_t* next_sibling_hdr(const hdr_field_t* const hf);/** not used yet */hdr_field_t* get_hdr_by_name(const sip_msg_t* const msg, const char* const name, const int name_len);hdr_field_t* next_sibling_hdr_by_name(const hdr_field_t* const hf);int set_path_vector(struct sip_msg* msg, str* path);void reset_path_vector(struct sip_msg* const msg);int set_instance(struct sip_msg* msg, str* instance);void reset_instance(struct sip_msg* const msg);int set_ruid(struct sip_msg* msg, str* ruid);void reset_ruid(struct sip_msg* const msg);int set_ua(struct sip_msg* msg, str *location_ua);void reset_ua(struct sip_msg* const msg);/** force a specific send socket for forwarding a request. * @param msg - sip msg. * @param fsocket - forced socket, pointer to struct socket_info, can be 0 (in *                  which case it's equivalent to reset_force_socket()). */#define set_force_socket(msg, fsocket) \	do { \		(msg)->force_send_socket=(fsocket); \		if ((msg)->force_send_socket) \			(msg)->fwd_send_flags.f |= SND_F_FORCE_SOCKET; \		else \			(msg)->fwd_send_flags.f &= ~SND_F_FORCE_SOCKET; \	} while (0)/** reset a previously forced send socket. */#define reset_force_socket(msg) set_force_socket(msg, 0)/** * struct to identify a msg context * - the pair of pid and message-id */typedef struct msg_ctx_id {	int pid;	int msgid;} msg_ctx_id_t;/** * set msg context id * - return: -1 on error; 0 - on set  */int msg_ctx_id_set(const sip_msg_t* const msg, msg_ctx_id_t* const mid);/** * check msg context id * - return: -1 on error; 0 - on no match; 1 - on match */int msg_ctx_id_match(const sip_msg_t* const msg, const msg_ctx_id_t* const mid);/** * set msg time value */int msg_set_time(sip_msg_t* const msg);/** * reset content of msg->ldv (msg_ldata_t structure) */void msg_ldata_reset(sip_msg_t*);#endif
 |