|
|
|
@ -48,7 +48,6 @@ CTranscoder::CTranscoder()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
m_bStopThread = false;
|
|
|
|
m_bStopThread = false;
|
|
|
|
m_pThread = NULL;
|
|
|
|
m_pThread = NULL;
|
|
|
|
m_Streams.reserve(12);
|
|
|
|
|
|
|
|
m_bConnected = false;
|
|
|
|
m_bConnected = false;
|
|
|
|
m_LastKeepaliveTime.Now();
|
|
|
|
m_LastKeepaliveTime.Now();
|
|
|
|
m_LastActivityTime.Now();
|
|
|
|
m_LastActivityTime.Now();
|
|
|
|
@ -62,25 +61,7 @@ CTranscoder::CTranscoder()
|
|
|
|
|
|
|
|
|
|
|
|
CTranscoder::~CTranscoder()
|
|
|
|
CTranscoder::~CTranscoder()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
// close all streams
|
|
|
|
Close();
|
|
|
|
m_Mutex.lock();
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
for ( int i = 0; i < m_Streams.size(); i++ )
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
delete m_Streams[i];
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
m_Streams.clear();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
m_Mutex.unlock();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// kill threads
|
|
|
|
|
|
|
|
m_bStopThread = true;
|
|
|
|
|
|
|
|
if ( m_pThread != NULL )
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
m_pThread->join();
|
|
|
|
|
|
|
|
delete m_pThread;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
@ -89,13 +70,13 @@ CTranscoder::~CTranscoder()
|
|
|
|
bool CTranscoder::Init(void)
|
|
|
|
bool CTranscoder::Init(void)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
bool ok;
|
|
|
|
bool ok;
|
|
|
|
|
|
|
|
|
|
|
|
// reset stop flag
|
|
|
|
// reset stop flag
|
|
|
|
m_bStopThread = false;
|
|
|
|
m_bStopThread = false;
|
|
|
|
|
|
|
|
|
|
|
|
// create server's IP
|
|
|
|
// create server's IP
|
|
|
|
m_Ip = g_Reflector.GetTranscoderIp();
|
|
|
|
m_Ip = g_Reflector.GetTranscoderIp();
|
|
|
|
|
|
|
|
|
|
|
|
// create our socket
|
|
|
|
// create our socket
|
|
|
|
ok = m_Socket.Open(TRANSCODER_PORT);
|
|
|
|
ok = m_Socket.Open(TRANSCODER_PORT);
|
|
|
|
if ( ok )
|
|
|
|
if ( ok )
|
|
|
|
@ -116,19 +97,19 @@ void CTranscoder::Close(void)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
// close socket
|
|
|
|
// close socket
|
|
|
|
m_Socket.Close();
|
|
|
|
m_Socket.Close();
|
|
|
|
|
|
|
|
|
|
|
|
// close all streams
|
|
|
|
// close all streams
|
|
|
|
m_Mutex.lock();
|
|
|
|
m_Mutex.lock();
|
|
|
|
{
|
|
|
|
{
|
|
|
|
for ( int i = 0; i < m_Streams.size(); i++ )
|
|
|
|
for ( auto it=m_Streams.begin(); it!=m_Streams.end(); it++ )
|
|
|
|
{
|
|
|
|
{
|
|
|
|
delete m_Streams[i];
|
|
|
|
delete *it;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
m_Streams.clear();
|
|
|
|
m_Streams.clear();
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
m_Mutex.unlock();
|
|
|
|
m_Mutex.unlock();
|
|
|
|
|
|
|
|
|
|
|
|
// kill threads
|
|
|
|
// kill threads
|
|
|
|
m_bStopThread = true;
|
|
|
|
m_bStopThread = true;
|
|
|
|
if ( m_pThread != NULL )
|
|
|
|
if ( m_pThread != NULL )
|
|
|
|
@ -156,13 +137,13 @@ void CTranscoder::Task(void)
|
|
|
|
CIp Ip;
|
|
|
|
CIp Ip;
|
|
|
|
uint16 StreamId;
|
|
|
|
uint16 StreamId;
|
|
|
|
uint16 Port;
|
|
|
|
uint16 Port;
|
|
|
|
|
|
|
|
|
|
|
|
// anything coming in from codec server ?
|
|
|
|
// anything coming in from codec server ?
|
|
|
|
//if ( (m_Socket.Receive(&Buffer, &Ip, 20) != -1) && (Ip == m_Ip) )
|
|
|
|
//if ( (m_Socket.Receive(&Buffer, &Ip, 20) != -1) && (Ip == m_Ip) )
|
|
|
|
if ( m_Socket.Receive(&Buffer, &Ip, 20) != -1 )
|
|
|
|
if ( m_Socket.Receive(&Buffer, &Ip, 20) != -1 )
|
|
|
|
{
|
|
|
|
{
|
|
|
|
m_LastActivityTime.Now();
|
|
|
|
m_LastActivityTime.Now();
|
|
|
|
|
|
|
|
|
|
|
|
// crack packet
|
|
|
|
// crack packet
|
|
|
|
if ( IsValidStreamDescrPacket(Buffer, &StreamId, &Port) )
|
|
|
|
if ( IsValidStreamDescrPacket(Buffer, &StreamId, &Port) )
|
|
|
|
{
|
|
|
|
{
|
|
|
|
@ -185,15 +166,15 @@ void CTranscoder::Task(void)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
m_bConnected = true;
|
|
|
|
m_bConnected = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// keep client alive
|
|
|
|
// keep client alive
|
|
|
|
if ( m_LastKeepaliveTime.DurationSinceNow() > TRANSCODER_KEEPALIVE_PERIOD )
|
|
|
|
if ( m_LastKeepaliveTime.DurationSinceNow() > TRANSCODER_KEEPALIVE_PERIOD )
|
|
|
|
{
|
|
|
|
{
|
|
|
|
//
|
|
|
|
//
|
|
|
|
HandleKeepalives();
|
|
|
|
HandleKeepalives();
|
|
|
|
|
|
|
|
|
|
|
|
// update time
|
|
|
|
// update time
|
|
|
|
m_LastKeepaliveTime.Now();
|
|
|
|
m_LastKeepaliveTime.Now();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
@ -205,9 +186,9 @@ void CTranscoder::Task(void)
|
|
|
|
CCodecStream *CTranscoder::GetStream(CPacketStream *PacketStream, uint8 uiCodecIn)
|
|
|
|
CCodecStream *CTranscoder::GetStream(CPacketStream *PacketStream, uint8 uiCodecIn)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
CBuffer Buffer;
|
|
|
|
CBuffer Buffer;
|
|
|
|
|
|
|
|
|
|
|
|
CCodecStream *stream = NULL;
|
|
|
|
CCodecStream *stream = NULL;
|
|
|
|
|
|
|
|
|
|
|
|
// do we need transcoding
|
|
|
|
// do we need transcoding
|
|
|
|
if ( uiCodecIn != CODEC_NONE )
|
|
|
|
if ( uiCodecIn != CODEC_NONE )
|
|
|
|
{
|
|
|
|
{
|
|
|
|
@ -217,17 +198,17 @@ CCodecStream *CTranscoder::GetStream(CPacketStream *PacketStream, uint8 uiCodecI
|
|
|
|
// yes, post openstream request
|
|
|
|
// yes, post openstream request
|
|
|
|
EncodeOpenstreamPacket(&Buffer, uiCodecIn, (uiCodecIn == CODEC_AMBEPLUS) ? CODEC_AMBE2PLUS : CODEC_AMBEPLUS);
|
|
|
|
EncodeOpenstreamPacket(&Buffer, uiCodecIn, (uiCodecIn == CODEC_AMBEPLUS) ? CODEC_AMBE2PLUS : CODEC_AMBEPLUS);
|
|
|
|
m_Socket.Send(Buffer, m_Ip, TRANSCODER_PORT);
|
|
|
|
m_Socket.Send(Buffer, m_Ip, TRANSCODER_PORT);
|
|
|
|
|
|
|
|
|
|
|
|
// wait relpy here
|
|
|
|
// wait relpy here
|
|
|
|
if ( m_SemaphoreOpenStream.WaitFor(AMBED_OPENSTREAM_TIMEOUT) )
|
|
|
|
if ( m_SemaphoreOpenStream.WaitFor(AMBED_OPENSTREAM_TIMEOUT) )
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if ( m_bStreamOpened )
|
|
|
|
if ( m_bStreamOpened )
|
|
|
|
{
|
|
|
|
{
|
|
|
|
std::cout << "ambed openstream ok" << std::endl;
|
|
|
|
std::cout << "ambed openstream ok" << std::endl;
|
|
|
|
|
|
|
|
|
|
|
|
// create stream object
|
|
|
|
// create stream object
|
|
|
|
stream = new CCodecStream(PacketStream, m_StreamidOpenStream, uiCodecIn, (uiCodecIn == CODEC_AMBEPLUS) ? CODEC_AMBE2PLUS : CODEC_AMBEPLUS);
|
|
|
|
stream = new CCodecStream(PacketStream, m_StreamidOpenStream, uiCodecIn, (uiCodecIn == CODEC_AMBEPLUS) ? CODEC_AMBE2PLUS : CODEC_AMBEPLUS);
|
|
|
|
|
|
|
|
|
|
|
|
// init it
|
|
|
|
// init it
|
|
|
|
if ( stream->Init(m_PortOpenStream) )
|
|
|
|
if ( stream->Init(m_PortOpenStream) )
|
|
|
|
{
|
|
|
|
{
|
|
|
|
@ -255,7 +236,7 @@ CCodecStream *CTranscoder::GetStream(CPacketStream *PacketStream, uint8 uiCodecI
|
|
|
|
{
|
|
|
|
{
|
|
|
|
std::cout << "ambed openstream timeout" << std::endl;
|
|
|
|
std::cout << "ambed openstream timeout" << std::endl;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return stream;
|
|
|
|
return stream;
|
|
|
|
@ -264,46 +245,46 @@ CCodecStream *CTranscoder::GetStream(CPacketStream *PacketStream, uint8 uiCodecI
|
|
|
|
void CTranscoder::ReleaseStream(CCodecStream *stream)
|
|
|
|
void CTranscoder::ReleaseStream(CCodecStream *stream)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
CBuffer Buffer;
|
|
|
|
CBuffer Buffer;
|
|
|
|
|
|
|
|
|
|
|
|
if ( stream != NULL )
|
|
|
|
if ( stream != NULL )
|
|
|
|
{
|
|
|
|
{
|
|
|
|
// look for the stream
|
|
|
|
// look for the stream
|
|
|
|
bool found = false;
|
|
|
|
bool found = false;
|
|
|
|
Lock();
|
|
|
|
Lock();
|
|
|
|
{
|
|
|
|
{
|
|
|
|
for ( int i = 0; (i < m_Streams.size()) && !found; i++ )
|
|
|
|
for ( auto it=m_Streams.begin(); it!=m_Streams.end(); it++ )
|
|
|
|
{
|
|
|
|
{
|
|
|
|
// compare object pointers
|
|
|
|
// compare object pointers
|
|
|
|
if ( (m_Streams[i]) == stream )
|
|
|
|
if ( (*it) == stream )
|
|
|
|
{
|
|
|
|
{
|
|
|
|
// send close packet
|
|
|
|
// send close packet
|
|
|
|
EncodeClosestreamPacket(&Buffer, m_Streams[i]->GetStreamId());
|
|
|
|
EncodeClosestreamPacket(&Buffer, (*it)->GetStreamId());
|
|
|
|
m_Socket.Send(Buffer, m_Ip, TRANSCODER_PORT);
|
|
|
|
m_Socket.Send(Buffer, m_Ip, TRANSCODER_PORT);
|
|
|
|
|
|
|
|
|
|
|
|
// display stats
|
|
|
|
// display stats
|
|
|
|
if ( m_Streams[i]->GetPingMin() >= 0.0 )
|
|
|
|
if ( (*it)->GetPingMin() >= 0.0 )
|
|
|
|
{
|
|
|
|
{
|
|
|
|
char sz[256];
|
|
|
|
char sz[256];
|
|
|
|
sprintf(sz, "ambed stats (ms) : %.1f/%.1f/%.1f",
|
|
|
|
sprintf(sz, "ambed stats (ms) : %.1f/%.1f/%.1f",
|
|
|
|
m_Streams[i]->GetPingMin() * 1000.0,
|
|
|
|
(*it)->GetPingMin() * 1000.0,
|
|
|
|
m_Streams[i]->GetPingAve() * 1000.0,
|
|
|
|
(*it)->GetPingAve() * 1000.0,
|
|
|
|
m_Streams[i]->GetPingMax() * 1000.0);
|
|
|
|
(*it)->GetPingMax() * 1000.0);
|
|
|
|
std::cout << sz << std::endl;
|
|
|
|
std::cout << sz << std::endl;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ( m_Streams[i]->GetTimeoutPackets() > 0 )
|
|
|
|
if ( (*it)->GetTimeoutPackets() > 0 )
|
|
|
|
{
|
|
|
|
{
|
|
|
|
char sz[256];
|
|
|
|
char sz[256];
|
|
|
|
sprintf(sz, "ambed %d of %d packets timed out",
|
|
|
|
sprintf(sz, "ambed %d of %d packets timed out",
|
|
|
|
m_Streams[i]->GetTimeoutPackets(),
|
|
|
|
(*it)->GetTimeoutPackets(),
|
|
|
|
m_Streams[i]->GetTotalPackets());
|
|
|
|
(*it)->GetTotalPackets());
|
|
|
|
std::cout << sz << std::endl;
|
|
|
|
std::cout << sz << std::endl;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// and close it
|
|
|
|
// and close it
|
|
|
|
m_Streams[i]->Close();
|
|
|
|
(*it)->Close();
|
|
|
|
delete m_Streams[i];
|
|
|
|
delete (*it);
|
|
|
|
m_Streams.erase(m_Streams.begin()+i);
|
|
|
|
m_Streams.erase(it);
|
|
|
|
found = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
@ -317,11 +298,11 @@ void CTranscoder::ReleaseStream(CCodecStream *stream)
|
|
|
|
void CTranscoder::HandleKeepalives(void)
|
|
|
|
void CTranscoder::HandleKeepalives(void)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
CBuffer keepalive;
|
|
|
|
CBuffer keepalive;
|
|
|
|
|
|
|
|
|
|
|
|
// send keepalive
|
|
|
|
// send keepalive
|
|
|
|
EncodeKeepAlivePacket(&keepalive);
|
|
|
|
EncodeKeepAlivePacket(&keepalive);
|
|
|
|
m_Socket.Send(keepalive, m_Ip, TRANSCODER_PORT);
|
|
|
|
m_Socket.Send(keepalive, m_Ip, TRANSCODER_PORT);
|
|
|
|
|
|
|
|
|
|
|
|
// check if still with us
|
|
|
|
// check if still with us
|
|
|
|
if ( m_bConnected && (m_LastActivityTime.DurationSinceNow() >= TRANSCODER_KEEPALIVE_TIMEOUT) )
|
|
|
|
if ( m_bConnected && (m_LastActivityTime.DurationSinceNow() >= TRANSCODER_KEEPALIVE_TIMEOUT) )
|
|
|
|
{
|
|
|
|
{
|
|
|
|
@ -337,7 +318,7 @@ void CTranscoder::HandleKeepalives(void)
|
|
|
|
bool CTranscoder::IsValidKeepAlivePacket(const CBuffer &Buffer)
|
|
|
|
bool CTranscoder::IsValidKeepAlivePacket(const CBuffer &Buffer)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
uint8 tag[] = { 'A','M','B','E','D','P','O','N','G' };
|
|
|
|
uint8 tag[] = { 'A','M','B','E','D','P','O','N','G' };
|
|
|
|
|
|
|
|
|
|
|
|
bool valid = false;
|
|
|
|
bool valid = false;
|
|
|
|
if ( (Buffer.size() == 9) && (Buffer.Compare(tag, sizeof(tag)) == 0) )
|
|
|
|
if ( (Buffer.size() == 9) && (Buffer.Compare(tag, sizeof(tag)) == 0) )
|
|
|
|
{
|
|
|
|
{
|
|
|
|
@ -349,7 +330,7 @@ bool CTranscoder::IsValidKeepAlivePacket(const CBuffer &Buffer)
|
|
|
|
bool CTranscoder::IsValidStreamDescrPacket(const CBuffer &Buffer, uint16 *Id, uint16 *Port)
|
|
|
|
bool CTranscoder::IsValidStreamDescrPacket(const CBuffer &Buffer, uint16 *Id, uint16 *Port)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
uint8 tag[] = { 'A','M','B','E','D','S','T','D' };
|
|
|
|
uint8 tag[] = { 'A','M','B','E','D','S','T','D' };
|
|
|
|
|
|
|
|
|
|
|
|
bool valid = false;
|
|
|
|
bool valid = false;
|
|
|
|
if ( (Buffer.size() == 14) && (Buffer.Compare(tag, sizeof(tag)) == 0) )
|
|
|
|
if ( (Buffer.size() == 14) && (Buffer.Compare(tag, sizeof(tag)) == 0) )
|
|
|
|
{
|
|
|
|
{
|
|
|
|
@ -365,7 +346,7 @@ bool CTranscoder::IsValidStreamDescrPacket(const CBuffer &Buffer, uint16 *Id, ui
|
|
|
|
bool CTranscoder::IsValidNoStreamAvailablePacket(const CBuffer&Buffer)
|
|
|
|
bool CTranscoder::IsValidNoStreamAvailablePacket(const CBuffer&Buffer)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
uint8 tag[] = { 'A','M','B','E','D','B','U','S','Y' };
|
|
|
|
uint8 tag[] = { 'A','M','B','E','D','B','U','S','Y' };
|
|
|
|
|
|
|
|
|
|
|
|
return ( (Buffer.size() == 9) && (Buffer.Compare(tag, sizeof(tag)) == 0) );
|
|
|
|
return ( (Buffer.size() == 9) && (Buffer.Compare(tag, sizeof(tag)) == 0) );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
@ -376,7 +357,7 @@ bool CTranscoder::IsValidNoStreamAvailablePacket(const CBuffer&Buffer)
|
|
|
|
void CTranscoder::EncodeKeepAlivePacket(CBuffer *Buffer)
|
|
|
|
void CTranscoder::EncodeKeepAlivePacket(CBuffer *Buffer)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
uint8 tag[] = { 'A','M','B','E','D','P','I','N','G' };
|
|
|
|
uint8 tag[] = { 'A','M','B','E','D','P','I','N','G' };
|
|
|
|
|
|
|
|
|
|
|
|
Buffer->Set(tag, sizeof(tag));
|
|
|
|
Buffer->Set(tag, sizeof(tag));
|
|
|
|
Buffer->Append((uint8 *)(const char *)g_Reflector.GetCallsign(), CALLSIGN_LEN);
|
|
|
|
Buffer->Append((uint8 *)(const char *)g_Reflector.GetCallsign(), CALLSIGN_LEN);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
@ -394,8 +375,7 @@ void CTranscoder::EncodeOpenstreamPacket(CBuffer *Buffer, uint8 uiCodecIn, uint8
|
|
|
|
void CTranscoder::EncodeClosestreamPacket(CBuffer *Buffer, uint16 uiStreamId)
|
|
|
|
void CTranscoder::EncodeClosestreamPacket(CBuffer *Buffer, uint16 uiStreamId)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
uint8 tag[] = { 'A','M','B','E','D','C','S' };
|
|
|
|
uint8 tag[] = { 'A','M','B','E','D','C','S' };
|
|
|
|
|
|
|
|
|
|
|
|
Buffer->Set(tag, sizeof(tag));
|
|
|
|
Buffer->Set(tag, sizeof(tag));
|
|
|
|
Buffer->Append((uint16)uiStreamId);
|
|
|
|
Buffer->Append((uint16)uiStreamId);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|