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

📄 rtpp_record.c

📁 RTP Proxy,用于Voip网络数据的代理传递
💻 C
字号:
/* * Copyright (c) 2004-2006 Maxim Sobolev <sobomax@FreeBSD.org> * Copyright (c) 2006-2007 Sippy Software, Inc., http://www.sippysoft.com * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright *    notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright *    notice, this list of conditions and the following disclaimer in the *    documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. * * $Id: rtpp_record.c,v 1.9 2008/03/18 02:26:29 sobomax Exp $ * */#include <sys/types.h>#include <sys/socket.h>#include <sys/stat.h>#include <sys/uio.h>#include <netinet/in.h>#include <errno.h>#include <fcntl.h>#include <limits.h>#include <stdio.h>#include <stdlib.h>#include <string.h>#include <unistd.h>#include "rtpp_log.h"#include "rtpp_record.h"#include "rtpp_session.h"#include "rtpp_util.h"enum record_mode {MODE_LOCAL_PKT, MODE_REMOTE_RTP}; /* MODE_LOCAL_RTP/MODE_REMOTE_PKT? */struct rtpp_record_channel {    char spath[PATH_MAX + 1];    char rpath[PATH_MAX + 1];    int fd;    int needspool;    char rbuf[4096];    int rbuf_len;    enum record_mode mode;};#define	RRC_CAST(x)	((struct rtpp_record_channel *)(x))void *ropen(struct cfg *cf, struct rtpp_session *sp, char *rname, int orig){    struct rtpp_record_channel *rrc;    const char *sdir;    char *cp, *tmp;    int n, port;    struct sockaddr_storage raddr;    rrc = malloc(sizeof(*rrc));    if (rrc == NULL) {	rtpp_log_ewrite(RTPP_LOG_ERR, sp->log, "can't allocate memory");	return NULL;    }    memset(rrc, 0, sizeof(*rrc));    if (rname != NULL && strncmp("udp:", rname, 4) == 0) {	tmp = strdup(rname + 4);	if (tmp == NULL) {	    rtpp_log_ewrite(RTPP_LOG_ERR, sp->log, "can't allocate memory");	    return NULL;	}	rrc->mode = MODE_REMOTE_RTP;	rrc->needspool = 0;	cp = strrchr(tmp, ':');	if (cp == NULL) {	    rtpp_log_write(RTPP_LOG_ERR, sp->log, "remote recording target specification should include port number");	    free(rrc);	    free(tmp);	    return NULL;	}	*cp = '\0';	cp++;	if (sp->rtcp == NULL) {	    /* Handle RTCP (increase target port by 1) */	    port = atoi(cp);	    if (port <= 0 || port > ((sp->rtcp != NULL) ? 65534 : 65535)) {		rtpp_log_write(RTPP_LOG_ERR, sp->log, "invalid port in the remote recording target specification");		free(rrc);		free(tmp);		return NULL;	    }	    sprintf(cp, "%d", port + 1);	}	n = resolve(sstosa(&raddr), AF_INET, tmp, cp, AI_PASSIVE);	if (n != 0) {	    rtpp_log_write(RTPP_LOG_ERR, sp->log, "ropen: getaddrinfo: %s", gai_strerror(n));	    free(rrc);	    free(tmp);	    return NULL;	}	rrc->fd = socket(AF_INET, SOCK_DGRAM, 0);	if (rrc->fd == -1) {	    rtpp_log_ewrite(RTPP_LOG_ERR, sp->log, "ropen: can't create socket");	    free(rrc);	    free(tmp);	    return NULL;	}	if (connect(rrc->fd, sstosa(&raddr), SA_LEN(sstosa(&raddr))) == -1) {	    rtpp_log_ewrite(RTPP_LOG_ERR, sp->log, "ropen: can't connect socket");	    close(rrc->fd);	    free(rrc);	    free(tmp);	    return NULL;	}	free(tmp);	return (void *)(rrc);    }    if (cf->rdir == NULL) {	rtpp_log_write(RTPP_LOG_ERR, sp->log, "directory for saving local recordings is not configured");	free(rrc);	return NULL;    }    if (cf->sdir == NULL) {	sdir = cf->rdir;	rrc->needspool = 0;    } else {	sdir = cf->sdir;	rrc->needspool = 1;	if (rname == NULL) {	    sprintf(rrc->rpath, "%s/%s=%s.%c.%s", cf->rdir, sp->call_id, sp->tag,	      (orig != 0) ? 'o' : 'a', (sp->rtcp != NULL) ? "rtp" : "rtcp");	} else {	    sprintf(rrc->rpath, "%s/%s.%s", cf->rdir, rname,	      (sp->rtcp != NULL) ? "rtp" : "rtcp");	}    }    if (rname == NULL) {	sprintf(rrc->spath, "%s/%s=%s.%c.%s", sdir, sp->call_id, sp->tag,	  (orig != 0) ? 'o' : 'a', (sp->rtcp != NULL) ? "rtp" : "rtcp");    } else {	sprintf(rrc->spath, "%s/%s.%s", sdir, rname,	  (sp->rtcp != NULL) ? "rtp" : "rtcp");    }    rrc->fd = open(rrc->spath, O_WRONLY | O_CREAT | O_TRUNC, DEFFILEMODE);    if (rrc->fd == -1) {	rtpp_log_ewrite(RTPP_LOG_ERR, sp->log, "can't open file %s for writing",	  rrc->spath);	free(rrc);	return NULL;    }    return (void *)(rrc);}static intflush_rbuf(struct rtpp_session *sp, void *rrc){    int rval;    rval = write(RRC_CAST(rrc)->fd, RRC_CAST(rrc)->rbuf, RRC_CAST(rrc)->rbuf_len);    if (rval != -1) {	RRC_CAST(rrc)->rbuf_len = 0;	return 0;    }    rtpp_log_ewrite(RTPP_LOG_ERR, sp->log, "error while recording session (%s)",      (sp->rtcp != NULL) ? "RTP" : "RTCP");    /* Prevent futher writing if error happens */    close(RRC_CAST(rrc)->fd);    RRC_CAST(rrc)->fd = -1;    return -1;}static intprepare_pkt_hdr(struct rtpp_session *sp, struct rtp_packet *packet, struct pkt_hdr *hdrp){    memset(hdrp, 0, sizeof(*hdrp));    hdrp->time = packet->rtime;    if (hdrp->time == -1) {	rtpp_log_ewrite(RTPP_LOG_ERR, sp->log, "can't get current time");	return -1;    }    switch (sstosa(&packet->raddr)->sa_family) {    case AF_INET:	hdrp->addr.in4.sin_family = sstosa(&packet->raddr)->sa_family;	hdrp->addr.in4.sin_port = satosin(&packet->raddr)->sin_port;	hdrp->addr.in4.sin_addr = satosin(&packet->raddr)->sin_addr;	break;    case AF_INET6:	hdrp->addr.in6.sin_family = sstosa(&packet->raddr)->sa_family;	hdrp->addr.in6.sin_port = satosin6(&packet->raddr)->sin6_port;	hdrp->addr.in6.sin_addr = satosin6(&packet->raddr)->sin6_addr;	break;    default:	abort();    }    hdrp->plen = packet->size;    return 0;}voidrwrite(struct rtpp_session *sp, void *rrc, struct rtp_packet *packet){    struct iovec v[2];    struct pkt_hdr hdr;    int rval;    if (RRC_CAST(rrc)->fd == -1)	return;    if (RRC_CAST(rrc)->mode == MODE_REMOTE_RTP) {	send(RRC_CAST(rrc)->fd, packet->buf, packet->size, 0);	return;    }    /* Check if the write buffer has necessary space, and flush if not */    if ((RRC_CAST(rrc)->rbuf_len + sizeof(struct pkt_hdr) + packet->size > sizeof(RRC_CAST(rrc)->rbuf)) && RRC_CAST(rrc)->rbuf_len > 0)	if (flush_rbuf(sp, rrc) != 0)	    return;    /* Check if received packet doesn't fit into the buffer, do synchronous write  if so*/    if (RRC_CAST(rrc)->rbuf_len + sizeof(struct pkt_hdr) + packet->size > sizeof(RRC_CAST(rrc)->rbuf)) {	if (prepare_pkt_hdr(sp, packet, &hdr) != 0)	    return;	v[0].iov_base = (void *)&hdr;	v[0].iov_len = sizeof(hdr);	v[1].iov_base = packet->buf;	v[1].iov_len = packet->size;	rval = writev(RRC_CAST(rrc)->fd, v, 2);	if (rval != -1)	    return;	rtpp_log_ewrite(RTPP_LOG_ERR, sp->log, "error while recording session (%s)",	  (sp->rtcp != NULL) ? "RTP" : "RTCP");	/* Prevent futher writing if error happens */	close(RRC_CAST(rrc)->fd);	RRC_CAST(rrc)->fd = -1;	return;    }    if (prepare_pkt_hdr(sp, packet, (struct pkt_hdr *)(RRC_CAST(rrc)->rbuf + RRC_CAST(rrc)->rbuf_len)) != 0)	return;    RRC_CAST(rrc)->rbuf_len += sizeof(struct pkt_hdr);    memcpy(RRC_CAST(rrc)->rbuf + RRC_CAST(rrc)->rbuf_len, packet->buf, packet->size);    RRC_CAST(rrc)->rbuf_len += packet->size;}voidrclose(struct rtpp_session *sp, void *rrc){    if (RRC_CAST(rrc)->mode != MODE_REMOTE_RTP && RRC_CAST(rrc)->rbuf_len > 0)	flush_rbuf(sp, rrc);    if (RRC_CAST(rrc)->fd != -1)	close(RRC_CAST(rrc)->fd);    if (RRC_CAST(rrc)->needspool == 1)	if (rename(RRC_CAST(rrc)->spath, RRC_CAST(rrc)->rpath) == -1)	    rtpp_log_ewrite(RTPP_LOG_ERR, sp->log, "can't move "	      "session record from spool into permanent storage");    free(rrc);}

⌨️ 快捷键说明

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