webrtc/modules/rtp_rtcp/source/ssrc_database.cc

313 lines
7.9 KiB
C++

/*
* Copyright (c) 2011 The WebRTC project authors. All Rights Reserved.
*
* Use of this source code is governed by a BSD-style license
* that can be found in the LICENSE file in the root of the source
* tree. An additional intellectual property rights grant can be found
* in the file PATENTS. All contributing project authors may
* be found in the AUTHORS file in the root of the source tree.
*/
#include "ssrc_database.h"
#include "critical_section_wrapper.h"
#include "trace.h"
#include <stdlib.h>
#include <cassert>
#ifdef _WIN32
#include <windows.h>
#include <MMSystem.h> //timeGetTime
#pragma warning(disable:4311)
#pragma warning(disable:4312)
// Platform SDK fixes when building with /Wp64 for a 32 bits target.
#if !defined(_WIN64) && defined(_Wp64)
#ifdef InterlockedExchangePointer
#undef InterlockedExchangePointer
// The problem is that the macro provided for InterlockedExchangePointer() is
// doing a (LONG) C-style cast that triggers invariably the warning C4312 when
// building on 32 bits.
inline void* InterlockedExchangePointer(void* volatile* target, void* value)
{
return reinterpret_cast<void*>(static_cast<LONG_PTR>(InterlockedExchange(
reinterpret_cast<volatile LONG*>(target),
static_cast<LONG>(reinterpret_cast<LONG_PTR>(value)))));
}
#endif // #ifdef InterlockedExchangePointer
#endif //!defined(_WIN64) && defined(_Wp64)
#else
#include <stdio.h>
#include <string.h>
#include <time.h>
#include <sys/time.h>
#ifndef WEBRTC_NO_AUTO_PTR // are we allowed to use auto_ptrs?
#include <memory> // definition of auto_ptr
#endif
#endif
namespace webrtc {
// Construct On First Use idiom. Avoids "static initialization order fiasco" (JFGI).
SSRCDatabase*&
SSRCDatabase::StaticInstance(SsrcDatabaseCount inc)
{
static volatile long theSSRCDatabaseCount = 0; // this needs to be long due to Windows, not an issue due to its usage
static SSRCDatabase* theSSRCDatabase = NULL;
SsrcDatabaseCreate state = kSsrcDbExist;
#ifndef _WIN32
#ifdef WEBRTC_NO_AUTO_PTR
// since we only have InterlockedExchange on windows and no auto_ptrs, this will result in a memory leak but we accept it for now
static CriticalSectionWrapper* crtiSect(CriticalSectionWrapper::CreateCriticalSection());
CriticalSectionScoped lock(*crtiSect);
#else
static std::auto_ptr<CriticalSectionWrapper> crtiSect = std::auto_ptr<CriticalSectionWrapper>(CriticalSectionWrapper::CreateCriticalSection());
CriticalSectionScoped lock(*crtiSect);
#endif
if(inc == kSsrcDbInc)
{
theSSRCDatabaseCount++;
if(theSSRCDatabaseCount == 1)
{
state = kSsrcDbCreate;
}
} else
{
theSSRCDatabaseCount--;
if(theSSRCDatabaseCount == 0)
{
state = kSsrcDbDestroy;
}
}
if(state == kSsrcDbCreate)
{
theSSRCDatabase = new SSRCDatabase();
}else if(state == kSsrcDbDestroy)
{
SSRCDatabase* oldValue = theSSRCDatabase;
theSSRCDatabase = NULL;
if(oldValue)
{
delete oldValue;
}
return theSSRCDatabase;
}
#else
// Windows
if(inc == kSsrcDbInc)
{
if(1 == InterlockedIncrement(&theSSRCDatabaseCount))
{
state = kSsrcDbCreate;
}
} else
{
int newValue = InterlockedDecrement(&theSSRCDatabaseCount);
if(newValue == 0)
{
state = kSsrcDbDestroy;
}
}
if(state == kSsrcDbCreate)
{
SSRCDatabase* newValue = new SSRCDatabase();
SSRCDatabase* oldValue = (SSRCDatabase*)InterlockedExchangePointer(reinterpret_cast<void* volatile*>(&theSSRCDatabase), newValue);
assert(oldValue == NULL);
}else if(state == kSsrcDbDestroy)
{
SSRCDatabase* oldValue = (SSRCDatabase*)InterlockedExchangePointer(reinterpret_cast<void* volatile*>(&theSSRCDatabase), NULL);
if(oldValue)
{
delete oldValue;
}
return theSSRCDatabase;
}
#endif
assert(theSSRCDatabase);
return theSSRCDatabase;
}
SSRCDatabase*
SSRCDatabase::GetSSRCDatabase()
{
return StaticInstance(kSsrcDbInc);
}
void
SSRCDatabase::ReturnSSRCDatabase()
{
StaticInstance(kSsrcDbDec);
}
WebRtc_UWord32
SSRCDatabase::CreateSSRC()
{
CriticalSectionScoped lock(*_critSect);
WebRtc_UWord32 ssrc = GenerateRandom();
#ifndef WEBRTC_NO_STL
while(_ssrcMap.find(ssrc) != _ssrcMap.end())
{
ssrc = GenerateRandom();
}
_ssrcMap[ssrc] = 0;
#else
if(_sizeOfSSRC <= _numberOfSSRC)
{
// allocate more space
const int newSize = _sizeOfSSRC + 10;
WebRtc_UWord32* tempSSRCVector = new WebRtc_UWord32[newSize];
memcpy(tempSSRCVector, _ssrcVector, _sizeOfSSRC*sizeof(WebRtc_UWord32));
delete [] _ssrcVector;
_ssrcVector = tempSSRCVector;
_sizeOfSSRC = newSize;
}
// check if in DB
if(_ssrcVector)
{
for (int i=0; i<_numberOfSSRC; i++)
{
if (_ssrcVector[i] == ssrc)
{
// we have a match
i = 0; // start over with a new ssrc
ssrc = GenerateRandom();
}
}
// add to database
_ssrcVector[_numberOfSSRC] = ssrc;
_numberOfSSRC++;
}
#endif
return ssrc;
}
WebRtc_Word32
SSRCDatabase::RegisterSSRC(const WebRtc_UWord32 ssrc)
{
CriticalSectionScoped lock(*_critSect);
#ifndef WEBRTC_NO_STL
_ssrcMap[ssrc] = 0;
#else
if(_sizeOfSSRC <= _numberOfSSRC)
{
// allocate more space
const int newSize = _sizeOfSSRC + 10;
WebRtc_UWord32* tempSSRCVector = new WebRtc_UWord32[newSize];
memcpy(tempSSRCVector, _ssrcVector, _sizeOfSSRC*sizeof(WebRtc_UWord32));
delete [] _ssrcVector;
_ssrcVector = tempSSRCVector;
_sizeOfSSRC = newSize;
}
// check if in DB
if(_ssrcVector)
{
for (int i=0; i<_numberOfSSRC; i++)
{
if (_ssrcVector[i] == ssrc)
{
// we have a match
return -1;
}
}
// add to database
_ssrcVector[_numberOfSSRC] = ssrc;
_numberOfSSRC++;
}
#endif
return 0;
}
WebRtc_Word32
SSRCDatabase::ReturnSSRC(const WebRtc_UWord32 ssrc)
{
CriticalSectionScoped lock(*_critSect);
#ifndef WEBRTC_NO_STL
_ssrcMap.erase(ssrc);
#else
if(_ssrcVector)
{
for (int i=0; i<_numberOfSSRC; i++)
{
if (_ssrcVector[i] == ssrc)
{
// we have a match
// remove from database
_ssrcVector[i] = _ssrcVector[_numberOfSSRC-1];
_numberOfSSRC--;
break;
}
}
}
#endif
return 0;
}
SSRCDatabase::SSRCDatabase()
{
// we need to seed the random generator, otherwise we get 26500 each time, hardly a random value :)
#ifdef _WIN32
srand(timeGetTime());
#else
struct timeval tv;
struct timezone tz;
gettimeofday(&tv, &tz);
srand(tv.tv_usec);
#endif
#ifdef WEBRTC_NO_STL
_sizeOfSSRC = 10;
_numberOfSSRC = 0;
_ssrcVector = new WebRtc_UWord32[10];
#endif
_critSect = CriticalSectionWrapper::CreateCriticalSection();
WEBRTC_TRACE(kTraceMemory, kTraceRtpRtcp, -1, "%s created", __FUNCTION__);
}
SSRCDatabase::~SSRCDatabase()
{
#ifdef WEBRTC_NO_STL
delete [] _ssrcVector;
#else
_ssrcMap.clear();
#endif
delete _critSect;
WEBRTC_TRACE(kTraceMemory, kTraceRtpRtcp, -1, "%s deleted", __FUNCTION__);
}
WebRtc_UWord32 SSRCDatabase::GenerateRandom()
{
WebRtc_UWord32 ssrc = 0;
do
{
ssrc = rand();
ssrc = ssrc <<16;
ssrc += rand();
} while (ssrc == 0 || ssrc == 0xffffffff);
return ssrc;
}
} // namespace webrtc