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/AmRtpAudio.cpp

223 lines
5.2 KiB

/*
* $Id: AmRtpAudio.cpp,v 1.1.2.7 2005/06/01 12:00:24 rco Exp $
*
* 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 "AmRtpAudio.h"
#include <sys/time.h>
#include <assert.h>
AmRtpAudio::AmRtpAudio(AmSession* _s)
: AmRtpStream(_s), AmAudio(0),
last_ts_i(false), use_default_plc(true),
send_only(false), playout_buffer(new AmPlayoutBuffer()),
last_check(0),last_check_i(false),send_int(false)
{
}
bool AmRtpAudio::checkInterval(unsigned int ts)
{
if(!last_check_i){
send_int = true;
last_check_i = true;
last_check = ts;
}
else {
if((ts - last_check) >= getFrameSize()){
send_int = true;
last_check = ts;
}
else {
send_int = false;
}
}
return send_int;
}
bool AmRtpAudio::sendIntReached()
{
return send_int;
}
/*
@param audio_buffer_ts [in] the current ts in the audio buffer
*/
int AmRtpAudio::receive(unsigned int audio_buffer_ts)
{
int size;
unsigned int ts;
while(true) {
size = AmRtpStream::receive((unsigned char*)samples,
(unsigned int)AUDIO_BUFFER_SIZE,ts,
audio_buffer_ts);
if(size <= 0)
break;
if(send_only){
last_ts_i = false;
continue;
}
if(!last_ts_i){
last_ts = ts;
last_ts_i = true;
}
if(ts_less()(last_ts,ts) && !begin_talk
&& (ts-last_ts <= PLC_MAX_SAMPLES)) {
int l_size = conceal_loss(ts - last_ts);
if(l_size>0){
playout_buffer->direct_write(last_ts,
(ShortSample*)samples.back_buffer(),
PCM16_B2S(l_size));
}
}
size = decode(size);
if(size <= 0){
ERROR("decode() returned %i\n",size);
return -1;
}
if(use_default_plc)
add_to_history(size);
playout_buffer->write(audio_buffer_ts, ts,
(ShortSample*)((unsigned char*)samples),
PCM16_B2S(size));
last_ts = ts + PCM16_B2S(size);
}
return size;
}
int AmRtpAudio::get(unsigned int user_ts, unsigned char* buffer, unsigned int nb_samples)
{
int size = read(user_ts,PCM16_S2B(nb_samples));
memcpy(buffer,(unsigned char*)samples,size);
return size;
}
int AmRtpAudio::read(unsigned int user_ts, unsigned int size)
{
u_int32_t rlen =
playout_buffer
->read(user_ts,
(ShortSample*)((unsigned char*)samples),
PCM16_B2S(size));
return PCM16_S2B(rlen);
}
int AmRtpAudio::write(unsigned int user_ts, unsigned int size)
{
return send(user_ts,(unsigned char*)samples,size);
}
void AmRtpAudio::init(const SdpPayload* sdp_payload)
{
DBG("AmRtpAudio::init(...)\n");
AmRtpStream::init(sdp_payload);
fmt.reset(new AmAudioRtpFormat(int_payload, format_parameters));
amci_codec_t* codec = fmt->getCodec();
use_default_plc = !(codec && codec->plc);
}
unsigned int AmRtpAudio::conceal_loss(unsigned int ts_diff)
{
int s=0;
if(!use_default_plc){
amci_codec_t* codec = fmt->getCodec();
long h_codec = fmt->getHCodec();
assert(codec && codec->plc);
s = (*codec->plc)(samples.back_buffer(),PCM16_S2B(ts_diff),
fmt->channels,fmt->rate,h_codec);
DBG("codec specific PLC (ts_diff = %i; s = %i)\n",ts_diff,s);
}
else {
s = default_plc(samples.back_buffer(),PCM16_S2B(ts_diff),
fmt->channels,fmt->rate);
DBG("default PLC (ts_diff = %i; s = %i)\n",ts_diff,s);
}
return s;
}
unsigned int AmRtpAudio::default_plc(unsigned char* out_buf,
unsigned int size,
unsigned int channels,
unsigned int rate)
{
short* buf_offset = (short*)out_buf;
for(unsigned int i=0; i<(PCM16_B2S(size)/FRAMESZ); i++){
fec.dofe(buf_offset);
buf_offset += FRAMESZ;
}
return PCM16_S2B(buf_offset - (short*)out_buf);
}
void AmRtpAudio::add_to_history(unsigned int size)
{
short* buf_offset = (short*)((unsigned char*)samples);
for(unsigned int i=0; i<(PCM16_B2S(size)/FRAMESZ); i++){
fec.addtohistory(buf_offset);
buf_offset += FRAMESZ;
}
}
void AmRtpAudio::setAdaptivePlayout(bool on)
{
bool is_adaptive =
dynamic_cast<AmAdaptivePlayout*>
(playout_buffer.get()) != 0;
if(on == !is_adaptive){
if(is_adaptive){
playout_buffer.reset(new AmAdaptivePlayout());
DBG("Adaptive playout buffer activated\n");
}
else{
playout_buffer.reset(new AmPlayoutBuffer());
DBG("Adaptive playout buffer deactivated\n");
}
}
}