⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 reply.c

📁 用来作为linux中SIP SERVER,完成VOIP网络电话中服务器的功能
💻 C
字号:
/* * $Id: reply.c,v 1.28.2.1 2005/01/13 08:48:08 sobomax Exp $ * * Send a reply * * 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: *    info@iptel.org * * 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-18: buffer overflow patch committed (Jan on behalf of Maxim) * 2003-01-21: Errors reported via Error-Info header field - janakj * 2003-09-11: updated to new build_lump_rpl() interface (bogdan) * 2003-11-11: build_lump_rpl() removed, add_lump_rpl() has flags (bogdan) */#include <stdio.h>#include "../../ut.h"#include "../../parser/msg_parser.h"#include "../../data_lump_rpl.h"#include "../usrloc/usrloc.h"#include "rerrno.h"#include "reg_mod.h"#include "regtime.h"#include "reply.h"#define MAX_CONTACT_BUFFER 1024#define E_INFO "P-Registrar-Error: "#define E_INFO_LEN (sizeof(E_INFO) - 1)#define CONTACT_BEGIN "Contact: "#define CONTACT_BEGIN_LEN (sizeof(CONTACT_BEGIN) - 1)#define Q_PARAM ";q="#define Q_PARAM_LEN (sizeof(Q_PARAM) - 1)#define EXPIRES_PARAM ";expires="#define EXPIRES_PARAM_LEN (sizeof(EXPIRES_PARAM) - 1)#define CONTACT_SEP ", "#define CONTACT_SEP_LEN (sizeof(CONTACT_SEP) - 1)/* * Buffer for Contact header field */static struct {	char* buf;	int buf_len;	int data_len;} contact = {0, 0, 0};/* * Calculate the length of buffer needed to * print contacts */static inline unsigned int calc_buf_len(ucontact_t* c){	unsigned int len;	int qlen;	len = 0;	while(c) {		if (VALID_CONTACT(c, act_time)) {			if (len) len += CONTACT_SEP_LEN;			len += 2 /* < > */ + c->c.len;			qlen = len_q(c->q);			if (qlen) len += Q_PARAM_LEN + qlen;			len += EXPIRES_PARAM_LEN + INT2STR_MAX_LEN;			if (c->received.s) {				len += 1 /* ; */ 					+ rcv_param.len 					+ 1 /* = */ 					+ 1 /* dquote */ 					+ c->received.len					+ 1 /* dquote */					;			}		}		c = c->next;	}	if (len) len += CONTACT_BEGIN_LEN + CRLF_LEN;	return len;}/* * Allocate a memory buffer and print Contact * header fields into it */int build_contact(ucontact_t* c){	char *p, *cp;	int fl, len;	contact.data_len = calc_buf_len(c);	if (!contact.data_len) return 0;	if (!contact.buf || (contact.buf_len < contact.data_len)) {		if (contact.buf) pkg_free(contact.buf);		contact.buf = (char*)pkg_malloc(contact.data_len);		if (!contact.buf) {			contact.data_len = 0;			contact.buf_len = 0;			LOG(L_ERR, "build_contact(): No memory left\n");			return -1;		} else {			contact.buf_len = contact.data_len;		}	}	p = contact.buf;		memcpy(p, CONTACT_BEGIN, CONTACT_BEGIN_LEN);	p += CONTACT_BEGIN_LEN;	fl = 0;	while(c) {		if (VALID_CONTACT(c, act_time)) {			if (fl) {				memcpy(p, CONTACT_SEP, CONTACT_SEP_LEN);				p += CONTACT_SEP_LEN;			} else {				fl = 1;			}			*p++ = '<';			memcpy(p, c->c.s, c->c.len);			p += c->c.len;			*p++ = '>';			len = len_q(c->q);			if (len) {				memcpy(p, Q_PARAM, Q_PARAM_LEN);				p += Q_PARAM_LEN;				memcpy(p, q2str(c->q, 0), len);				p += len;			}			memcpy(p, EXPIRES_PARAM, EXPIRES_PARAM_LEN);			p += EXPIRES_PARAM_LEN;			cp = int2str((int)(c->expires - act_time), &len);			memcpy(p, cp, len);			p += len;			if (c->received.s) {				*p++ = ';';				memcpy(p, rcv_param.s, rcv_param.len);				p += rcv_param.len;				*p++ = '=';				*p++ = '\"';				memcpy(p, c->received.s, c->received.len);				p += c->received.len;				*p++ = '\"';			}		}		c = c->next;	}	memcpy(p, CRLF, CRLF_LEN);	p += CRLF_LEN;	contact.data_len = p - contact.buf;	DBG("build_contact(): Created Contact HF: %.*s\n", contact.data_len, contact.buf);	return 0;}#define MSG_200 "OK"#define MSG_400 "Bad Request"#define MSG_500 "Server Internal Error"#define MSG_503 "Service Unavailable"#define EI_R_FINE       "No problem"                                /* R_FINE */#define EI_R_UL_DEL_R   "usrloc_record_delete failed"               /* R_UL_DEL_R */#define	EI_R_UL_GET_R   "usrloc_record_get failed"                  /* R_UL_GET */#define	EI_R_UL_NEW_R   "usrloc_record_new failed"                  /* R_UL_NEW_R */#define	EI_R_INV_CSEQ   "Invalid CSeq number"                       /* R_INV_CSEQ */#define	EI_R_UL_INS_C   "usrloc_contact_insert failed"              /* R_UL_INS_C */#define	EI_R_UL_INS_R   "usrloc_record_insert failed"               /* R_UL_INS_R */#define	EI_R_UL_DEL_C   "usrloc_contact_delete failed"              /* R_UL_DEL_C */#define	EI_R_UL_UPD_C   "usrloc_contact_update failed"              /* R_UL_UPD_C */#define	EI_R_TO_USER    "No username in To URI"                     /* R_TO_USER */#define	EI_R_AOR_LEN    "Address Of Record too long"                /* R_AOR_LEN */#define	EI_R_AOR_PARSE  "Error while parsing AOR"                   /* R_AOR_PARSE */#define	EI_R_INV_EXP    "Invalid expires param in contact"          /* R_INV_EXP */#define	EI_R_INV_Q      "Invalid q param in contact"                /* R_INV_Q */#define	EI_R_PARSE      "Message parse error"                       /* R_PARSE */#define	EI_R_TO_MISS    "To header not found"                       /* R_TO_MISS */#define	EI_R_CID_MISS   "Call-ID header not found"                  /* R_CID_MISS */#define	EI_R_CS_MISS    "CSeq header not found"                     /* R_CS_MISS */ #define	EI_R_PARSE_EXP	"Expires parse error"                       /* R_PARSE_EXP */#define	EI_R_PARSE_CONT	"Contact parse error"                       /* R_PARSE_CONT */#define	EI_R_STAR_EXP	"* used in contact and expires is not zero" /* R_STAR__EXP */#define	EI_R_STAR_CONT	"* used in contact and more than 1 contact" /* R_STAR_CONT */#define	EI_R_OOO	"Out of order request"                      /* R_OOO */#define	EI_R_RETRANS	"Retransmission"                            /* R_RETRANS */#define EI_R_UNESCAPE   "Error while unescaping username"           /* R_UNESCAPE */#define EI_R_TOO_MANY   "Too many registered contacts"              /* R_TOO_MANY */str error_info[] = {	{EI_R_FINE,       sizeof(EI_R_FINE) - 1},	{EI_R_UL_DEL_R,   sizeof(EI_R_UL_DEL_R) - 1},	{EI_R_UL_GET_R,   sizeof(EI_R_UL_GET_R) - 1},	{EI_R_UL_NEW_R,   sizeof(EI_R_UL_NEW_R) - 1},	{EI_R_INV_CSEQ,   sizeof(EI_R_INV_CSEQ) - 1},	{EI_R_UL_INS_C,   sizeof(EI_R_UL_INS_C) - 1},	{EI_R_UL_INS_R,   sizeof(EI_R_UL_INS_R) - 1},	{EI_R_UL_DEL_C,   sizeof(EI_R_UL_DEL_C) - 1},	{EI_R_UL_UPD_C,   sizeof(EI_R_UL_UPD_C) - 1},	{EI_R_TO_USER,    sizeof(EI_R_TO_USER) - 1},	{EI_R_AOR_LEN,    sizeof(EI_R_AOR_LEN) - 1},	{EI_R_AOR_PARSE,  sizeof(EI_R_AOR_PARSE) - 1},	{EI_R_INV_EXP,    sizeof(EI_R_INV_EXP) - 1},	{EI_R_INV_Q,      sizeof(EI_R_INV_Q) - 1},	{EI_R_PARSE,      sizeof(EI_R_PARSE) - 1},	{EI_R_TO_MISS,    sizeof(EI_R_TO_MISS) - 1},	{EI_R_CID_MISS,   sizeof(EI_R_CID_MISS) - 1},	{EI_R_CS_MISS,    sizeof(EI_R_CS_MISS) - 1},	{EI_R_PARSE_EXP,  sizeof(EI_R_PARSE_EXP) - 1},	{EI_R_PARSE_CONT, sizeof(EI_R_PARSE_CONT) - 1},	{EI_R_STAR_EXP,   sizeof(EI_R_STAR_EXP) - 1},	{EI_R_STAR_CONT,  sizeof(EI_R_STAR_CONT) - 1},	{EI_R_OOO,        sizeof(EI_R_OOO) - 1},	{EI_R_RETRANS,    sizeof(EI_R_RETRANS) - 1},	{EI_R_UNESCAPE,   sizeof(EI_R_UNESCAPE) - 1},	{EI_R_TOO_MANY,   sizeof(EI_R_TOO_MANY) - 1}};int codes[] = {	200, /* R_FINE */     	500, /* R_UL_DEL_R */	500, /* R_UL_GET */	500, /* R_UL_NEW_R */	400, /* R_INV_CSEQ */	500, /* R_UL_INS_C */	500, /* R_UL_INS_R */	500, /* R_UL_DEL_C */	500, /* R_UL_UPD_C */	400, /* R_TO_USER */	500, /* R_AOR_LEN */	400, /* R_AOR_PARSE */	400, /* R_INV_EXP */	400, /* R_INV_Q */	400, /* R_PARSE */	400, /* R_TO_MISS */	400, /* R_CID_MISS */	400, /* R_CS_MISS */	400, /* R_PARSE_EXP */	400, /* R_PARSE_CONT */	400, /* R_STAR_EXP */	400, /* R_STAR_CONT */	200, /* R_OOO */	200, /* R_RETRANS */	400, /* R_UNESCAPE */	503  /* R_TOO_MANY */};#define RETRY_AFTER "Retry-After: "#define RETRY_AFTER_LEN (sizeof(RETRY_AFTER) - 1)static int add_retry_after(struct sip_msg* _m){	char* buf, *ra_s; 	int ra_len; 	 	ra_s = int2str(retry_after, &ra_len); 	buf = (char*)pkg_malloc(RETRY_AFTER_LEN + ra_len + CRLF_LEN); 	if (!buf) { 		LOG(L_ERR, "add_retry_after: No memory left\n"); 		return -1; 	} 	memcpy(buf, RETRY_AFTER, RETRY_AFTER_LEN); 	memcpy(buf + RETRY_AFTER_LEN, ra_s, ra_len); 	memcpy(buf + RETRY_AFTER_LEN + ra_len, CRLF, CRLF_LEN); 	add_lump_rpl(_m, buf, RETRY_AFTER_LEN + ra_len + CRLF_LEN, 		     LUMP_RPL_HDR | LUMP_RPL_NODUP); 	return 0;}/* * Send a reply */int send_reply(struct sip_msg* _m){	long code;	char* msg = MSG_200; /* makes gcc shut up */	char* buf;	if (contact.data_len > 0) {		add_lump_rpl( _m, contact.buf, contact.data_len, LUMP_RPL_HDR|LUMP_RPL_NODUP|LUMP_RPL_NOFREE);		contact.data_len = 0;	}	code = codes[rerrno];	switch(code) {	case 200: msg = MSG_200; break;	case 400: msg = MSG_400; break;	case 500: msg = MSG_500; break;	case 503: msg = MSG_503; break;	}		if (code != 200) {		buf = (char*)pkg_malloc(E_INFO_LEN + error_info[rerrno].len + CRLF_LEN + 1);		if (!buf) {			LOG(L_ERR, "send_reply(): No memory left\n");			return -1;		}		memcpy(buf, E_INFO, E_INFO_LEN);		memcpy(buf + E_INFO_LEN, error_info[rerrno].s, error_info[rerrno].len);		memcpy(buf + E_INFO_LEN + error_info[rerrno].len, CRLF, CRLF_LEN);		add_lump_rpl( _m, buf, E_INFO_LEN + error_info[rerrno].len + CRLF_LEN,			LUMP_RPL_HDR|LUMP_RPL_NODUP);		if (code >= 500 && code < 600 && retry_after) {			if (add_retry_after(_m) < 0) {				return -1;			}		}	}	if (sl_reply(_m, (char*)code, msg) == -1) {		LOG(L_ERR, "send_reply(): Error while sending %ld %s\n", code, msg);		return -1;	} else return 0;	}/* * Release contact buffer if any */void free_contact_buf(void){	if (contact.buf) {		pkg_free(contact.buf);		contact.buf = 0;		contact.buf_len = 0;		contact.data_len = 0;	}}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -