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

498 lines
13 KiB

/*
* 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. This program is released under
* the GPL with the additional exemption that compiling, linking,
* and/or using OpenSSL is allowed.
*
* For a license to use the SEMS 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>
#include "AmSession.h"
#include "AmPlayoutBuffer.h"
AmAudioRtpFormat::AmAudioRtpFormat()
: AmAudioFormat(-1),
advertized_rate(-1)
{
}
AmAudioRtpFormat::~AmAudioRtpFormat()
{
}
int AmAudioRtpFormat::setCurrentPayload(Payload pl)
{
if (this->codec_id != pl.codec_id) {
DBG("setCurrentPayload({%u, '%s', %u, %u, %u, '%s'})\n",
pl.pt, pl.name.c_str(), pl.clock_rate, pl.advertised_clock_rate,
pl.codec_id, pl.format_parameters.c_str());
this->codec_id = pl.codec_id;
DBG("fmt.codec_id = %d", this->codec_id);
this->channels = 1;
this->rate = pl.clock_rate;
DBG("fmt.rate = %d", this->rate);
this->advertized_rate = pl.advertised_clock_rate;
DBG("fmt.advertized_rate = %d", this->advertized_rate);
this->frame_size = 20*this->rate/1000;
this->sdp_format_parameters = pl.format_parameters;
DBG("fmt.sdp_format_parameters = %s", this->sdp_format_parameters.c_str());
if (this->codec != NULL) {
destroyCodec();
}
}
return 0;
}
void AmAudioRtpFormat::initCodec()
{
amci_codec_fmt_info_t* fmt_i = NULL;
sdp_format_parameters_out = NULL; // reset
if( codec && codec->init ) {
if ((h_codec = (*codec->init)(sdp_format_parameters.c_str(),
&sdp_format_parameters_out, &fmt_i)) == -1) {
ERROR("could not initialize codec %i\n",codec->id);
} else {
if (NULL != sdp_format_parameters_out) {
DBG("negotiated fmt parameters '%s'\n", sdp_format_parameters_out);
log_demangled_stacktrace(L_DBG, 30);
}
if (NULL != fmt_i) {
unsigned int i=0;
while (i<4 && fmt_i[i].id) {
switch (fmt_i[i].id) {
case AMCI_FMT_FRAME_LENGTH : {
//frame_length=fmt_i[i].value; // ignored
} break;
case AMCI_FMT_FRAME_SIZE: {
frame_size=fmt_i[i].value;
} break;
case AMCI_FMT_ENCODED_FRAME_SIZE: {
// frame_encoded_size=fmt_i[i].value; // ignored
} break;
default: {
DBG("Unknown codec format descriptor: %d\n", fmt_i[i].id);
} break;
}
i++;
}
}
}
}
}
AmRtpAudio::AmRtpAudio(AmSession* _s, int _if)
: AmRtpStream(_s,_if), AmAudio(0),
/*last_ts_i(false),*/ use_default_plc(true),
playout_buffer(NULL),
m_playout_type(SIMPLE_PLAYOUT),
last_check(0),last_check_i(false),send_int(false),
last_send_ts_i(false)
{
#ifdef USE_SPANDSP_PLC
plc_state = plc_init(NULL);
#endif // USE_SPANDSP_PLC
}
AmRtpAudio::~AmRtpAudio() {
#ifdef USE_SPANDSP_PLC
plc_release(plc_state);
#endif // USE_SPANDSP_PLC
}
bool AmRtpAudio::checkInterval(unsigned long long ts)
{
if(!last_check_i){
send_int = true;
last_check_i = true;
last_check = ts;
}
else {
if(scaleSystemTS(ts - last_check) >= getFrameSize()){
send_int = true;
last_check = ts;
}
else {
send_int = false;
}
}
return send_int;
}
bool AmRtpAudio::sendIntReached()
{
return send_int;
}
bool AmRtpAudio::sendIntReached(unsigned long long ts)
{
if (!last_send_ts_i) return true;
else return (scaleSystemTS(ts - last_send_ts) >= getFrameSize());
}
unsigned int AmRtpAudio::bytes2samples(unsigned int bytes) const
{
return AmAudio::bytes2samples(bytes);
}
/*
@param wallclock_ts [in] the current ts in the audio buffer
*/
int AmRtpAudio::receive(unsigned long long system_ts)
{
int size;
unsigned int rtp_ts;
int new_payload = -1;
if(!fmt.get() || (!playout_buffer.get())) {
DBG("audio format not initialized\n");
return RTP_ERROR;
}
unsigned int wallclock_ts = scaleSystemTS(system_ts);
while(true) {
size = AmRtpStream::receive((unsigned char*)samples,
(unsigned int)AUDIO_BUFFER_SIZE, rtp_ts,
new_payload);
if(size <= 0) {
switch(size){
case 0: break;
case RTP_DTMF:
case RTP_UNKNOWN_PL:
case RTP_PARSE_ERROR:
continue;
case RTP_TIMEOUT:
//FIXME: postRequest(new SchedRequest(AmMediaProcessor::RemoveSession,s));
// post to the session (FIXME: is session always set? seems to be...)
session->postEvent(new AmRtpTimeoutEvent());
return -1;
case RTP_BUFFER_SIZE:
default:
ERROR("AmRtpStream::receive() returned %i\n",size);
//FIXME: postRequest(new SchedRequest(AmMediaProcessor::ClearSession,s));
// or AmMediaProcessor::instance()->clearSession(session);
return -1;
break;
}
break;
}
if (// don't process if we don't need to
// ignore CN
COMFORT_NOISE_PAYLOAD_TYPE == new_payload ||
// ignore packet if payload not found
setCurrentPayload(new_payload)
){
playout_buffer->clearLastTs();
continue;
}
size = decode(size);
if(size <= 0){
ERROR("decode() returned %i\n",size);
return -1;
}
// This only works because the possible ratio (Rate/TSRate)
// is 2. Rate and TSRate are only different in case of g722.
// For g722, TSRate=8000 and Rate=16000
//
AmAudioRtpFormat* rtp_fmt = (AmAudioRtpFormat*)fmt.get();
unsigned long long adjusted_rtp_ts = rtp_ts;
if(rtp_fmt->getRate() != rtp_fmt->getTSRate()) {
adjusted_rtp_ts =
(unsigned long long)rtp_ts *
(unsigned long long)rtp_fmt->getRate()
/ (unsigned long long)rtp_fmt->getTSRate();
}
playout_buffer->write(wallclock_ts, adjusted_rtp_ts,
(ShortSample*)((unsigned char *)samples),
PCM16_B2S(size), begin_talk);
if(!active) {
DBG("switching to active-mode\t(ts=%u;stream=%p)\n",
rtp_ts,this);
active = true;
}
}
return size;
}
int AmRtpAudio::get(unsigned long long system_ts, unsigned char* buffer,
int output_sample_rate, unsigned int nb_samples)
{
if (!(receiving || getPassiveMode())) return 0; // like nothing received
int ret = receive(system_ts);
if(ret < 0)
return ret; // like nothing received?
if (!active) return 0;
unsigned int user_ts = scaleSystemTS(system_ts);
nb_samples = (unsigned int)((float)nb_samples * (float)getSampleRate()
/ (float)output_sample_rate);
u_int32_t size =
PCM16_S2B(playout_buffer->read(user_ts,
(ShortSample*)((unsigned char*)samples),
nb_samples));
if(output_sample_rate != getSampleRate()) {
size = resampleOutput((unsigned char*)samples, size,
getSampleRate(), output_sample_rate);
}
memcpy(buffer,(unsigned char*)samples,size);
return size;
}
int AmRtpAudio::put(unsigned long long system_ts, unsigned char* buffer,
int input_sample_rate, unsigned int size)
{
last_send_ts_i = true;
last_send_ts = system_ts;
if(!size){
return 0;
}
if (mute) return 0;
memcpy((unsigned char*)samples,buffer,size);
size = resampleInput((unsigned char*)samples, size,
input_sample_rate, getSampleRate());
int s = encode(size);
if(s<=0){
return s;
}
AmAudioRtpFormat* rtp_fmt = (AmAudioRtpFormat*)fmt.get();
// pre-division by 100 is important
// so that the first multiplication
// does not overflow the 64bit int
unsigned long long user_ts =
system_ts * ((unsigned long long)rtp_fmt->getTSRate() / 100)
/ (WALLCLOCK_RATE/100);
return send((unsigned int)user_ts,(unsigned char*)samples,s);
}
void AmRtpAudio::getSdpOffer(unsigned int index, SdpMedia& offer)
{
if (offer.type != MT_AUDIO) return;
AmRtpStream::getSdpOffer(index,offer);
}
void AmRtpAudio::getSdpAnswer(unsigned int index,
const SdpMedia& offer,
SdpMedia& answer)
{
answer.type = MT_AUDIO;
AmRtpStream::getSdpAnswer(index,offer,answer);
}
int AmRtpAudio::init(const AmSdp& local,
const AmSdp& remote, bool force_symmetric_rtp)
{
DBG("AmRtpAudio::init(...)\n");
if(AmRtpStream::init(local,remote,force_symmetric_rtp)){
return -1;
}
AmAudioRtpFormat* fmt_p = new AmAudioRtpFormat();
PayloadMappingTable::iterator pl_it = pl_map.find(payload);
if ((pl_it == pl_map.end()) || (pl_it->second.remote_pt < 0)) {
DBG("no default payload has been set\n");
return -1;
}
fmt_p->setCurrentPayload(payloads[pl_it->second.index]);
fmt.reset(fmt_p);
amci_codec_t* codec = fmt->getCodec();
use_default_plc = ((codec==NULL) || (codec->plc == NULL));
fec.reset(new LowcFE(getSampleRate()));
if (m_playout_type == SIMPLE_PLAYOUT) {
playout_buffer.reset(new AmPlayoutBuffer(this,getSampleRate()));
} else if (m_playout_type == ADAPTIVE_PLAYOUT) {
playout_buffer.reset(new AmAdaptivePlayout(this,getSampleRate()));
} else {
playout_buffer.reset(new AmJbPlayout(this,getSampleRate()));
}
return 0;
}
unsigned int AmRtpAudio::getFrameSize()
{
if (!fmt.get())
return 0;
return ((AmAudioRtpFormat*)fmt.get())->getFrameSize();
}
int AmRtpAudio::setCurrentPayload(int payload)
{
if(payload != this->payload){
PayloadMappingTable::iterator pmt_it = pl_map.find(payload);
if(pmt_it == pl_map.end()){
ERROR("Could not set current payload: payload %i unknown to this stream\n",payload);
return -1;
}
unsigned char index = pmt_it->second.index;
if(index >= payloads.size()){
ERROR("Could not set current payload: payload %i maps to invalid index %i\n",
payload, index);
return -1;
}
this->payload = payload;
int res = ((AmAudioRtpFormat*)fmt.get())->setCurrentPayload(payloads[index]);
amci_codec_t* codec = fmt->getCodec();
use_default_plc = ((codec==NULL) || (codec->plc == NULL));
return res;
}
else {
return 0;
}
}
//int AmRtpAudio::getCurrentPayload() {
//return ((AmAudioRtpFormat *) fmt.get())->getCurrentPayload();
//}
unsigned int AmRtpAudio::conceal_loss(unsigned int ts_diff, unsigned char *buffer)
{
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)(buffer, PCM16_S2B(ts_diff),
fmt->channels,getSampleRate(),h_codec);
DBG("codec specific PLC (ts_diff = %i; s = %i)\n",ts_diff,s);
}
else {
s = default_plc(buffer, PCM16_S2B(ts_diff),
fmt->channels,getSampleRate());
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 sample_rate)
{
short* buf_offset = (short*)out_buf;
#ifdef USE_SPANDSP_PLC
plc_fillin(plc_state, buf_offset, PCM16_B2S(size));
#else
for(unsigned int i=0; i<(PCM16_B2S(size)/FRAMESZ); i++){
fec->dofe(buf_offset);
buf_offset += FRAMESZ;
}
#endif // USE_SPANDSP_PLC
return PCM16_S2B(buf_offset - (short*)out_buf);
}
void AmRtpAudio::add_to_history(int16_t *buffer, unsigned int size)
{
if (!use_default_plc)
return;
#ifdef USE_SPANDSP_PLC
plc_rx(plc_state, buffer, PCM16_B2S(size));
#else // USE_SPANDSP_PLC
int16_t* buf_offset = buffer;
unsigned int sample_rate = getSampleRate();
for(unsigned int i=0; i<(PCM16_B2S(size)/FRAMESZ); i++){
fec->addtohistory(buf_offset);
buf_offset += FRAMESZ;
}
#endif // USE_SPANDSP_PLC
}
void AmRtpAudio::setPlayoutType(PlayoutType type)
{
if (m_playout_type != type)
{
if (type == ADAPTIVE_PLAYOUT) {
session->lockAudio();
m_playout_type = type;
if (fmt.get())
playout_buffer.reset(new AmAdaptivePlayout(this,getSampleRate()));
session->unlockAudio();
DBG("Adaptive playout buffer activated\n");
}
else if (type == JB_PLAYOUT) {
session->lockAudio();
m_playout_type = type;
if (fmt.get())
playout_buffer.reset(new AmJbPlayout(this,getSampleRate()));
session->unlockAudio();
DBG("Adaptive jitter buffer activated\n");
}
else {
session->lockAudio();
m_playout_type = type;
if (fmt.get())
playout_buffer.reset(new AmPlayoutBuffer(this,getSampleRate()));
session->unlockAudio();
DBG("Simple playout buffer activated\n");
}
}
}