You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
sems/core/AmRtpReceiver.cpp

156 lines
3.3 KiB

/*
* $Id$
*
* Copyright (C) 2002-2003 Fhg Fokus
*
* This file is part of sems, a free SIP media server.
*
* sems 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
*
* sems 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
*/
#include "AmRtpReceiver.h"
#include "AmRtpStream.h"
#include "AmRtpPacket.h"
#include "log.h"
#include <errno.h>
#include <strings.h>
#include <sys/time.h>
#include <sys/poll.h>
#define MAX_RTP_SESSIONS 2048
#define RTP_POLL_TIMEOUT 50 /*50 ms*/
AmRtpReceiver* AmRtpReceiver::_instance=0;
AmRtpReceiver* AmRtpReceiver::instance()
{
if(!_instance)
_instance = new AmRtpReceiver();
return _instance;
}
AmRtpReceiver::AmRtpReceiver()
{
fds = new struct pollfd[MAX_RTP_SESSIONS];
nfds = 0;
}
AmRtpReceiver::~AmRtpReceiver()
{
delete [] (fds);
}
void AmRtpReceiver::on_stop()
{
}
void AmRtpReceiver::run()
{
unsigned int tmp_nfds = 0;
struct pollfd* tmp_fds = new struct pollfd[MAX_RTP_SESSIONS];
AmRtpPacket p;
while(true){
fds_mut.lock();
tmp_nfds = nfds;
memcpy(tmp_fds,fds,nfds*sizeof(struct pollfd));
fds_mut.unlock();
int ret = poll(tmp_fds,tmp_nfds,RTP_POLL_TIMEOUT);
if(ret < 0)
ERROR("AmRtpReceiver: poll: %s\n",strerror(errno));
if(ret < 1)
continue;
for(unsigned int i=0; i<tmp_nfds; i++) {
if(!(tmp_fds[i].revents & POLLIN))
continue;
if(p.recv(tmp_fds[i].fd) > 0){
int parse_res = p.parse();
gettimeofday(&p.recv_time,NULL);
streams_mut.lock();
Streams::iterator it = streams.find(tmp_fds[i].fd);
if(it != streams.end()) {
if (parse_res == -1) {
DBG("error while parsing RTP packet.\n");
it->second->clearRTPTimeout(&p.recv_time);
} else {
it->second->bufferPacket(&p);
}
}
streams_mut.unlock();
}
}
}
}
void AmRtpReceiver::addStream(int sd, AmRtpStream* stream)
{
fds_mut.lock();
if(nfds >= MAX_RTP_SESSIONS){
fds_mut.unlock();
ERROR("maximum number of sessions reached (%i)\n",
MAX_RTP_SESSIONS);
throw string("maximum number of sessions reached");
}
fds[nfds].fd = sd;
fds[nfds].events = POLLIN;
fds[nfds].revents = 0;
nfds++;
fds_mut.unlock();
streams_mut.lock();
streams.insert(std::make_pair(sd,stream));
streams_mut.unlock();
}
void AmRtpReceiver::removeStream(int sd)
{
fds_mut.lock();
for(unsigned int i=0; i<nfds; i++){
if(fds[i].fd == sd){
if(--nfds && (i < nfds)){
fds[i] = fds[nfds];
}
break;
}
}
fds_mut.unlock();
streams_mut.lock();
streams.erase(sd);
streams_mut.unlock();
}