
BUG=314 Review URL: https://webrtc-codereview.appspot.com/1279007 git-svn-id: http://webrtc.googlecode.com/svn/trunk@3777 4adac7df-926f-26a2-2b94-8c16560cd09d
218 lines
4.6 KiB
C++
218 lines
4.6 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
|
|
|
|
// TODO(hellner): investigate if it is necessary to disable these warnings.
|
|
#pragma warning(disable:4311)
|
|
#pragma warning(disable:4312)
|
|
#else
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <time.h>
|
|
#include <sys/time.h>
|
|
#endif
|
|
|
|
namespace webrtc {
|
|
SSRCDatabase*
|
|
SSRCDatabase::StaticInstance(CountOperation count_operation)
|
|
{
|
|
SSRCDatabase* impl =
|
|
GetStaticInstance<SSRCDatabase>(count_operation);
|
|
return impl;
|
|
}
|
|
|
|
SSRCDatabase*
|
|
SSRCDatabase::GetSSRCDatabase()
|
|
{
|
|
return StaticInstance(kAddRef);
|
|
}
|
|
|
|
void
|
|
SSRCDatabase::ReturnSSRCDatabase()
|
|
{
|
|
StaticInstance(kRelease);
|
|
}
|
|
|
|
uint32_t
|
|
SSRCDatabase::CreateSSRC()
|
|
{
|
|
CriticalSectionScoped lock(_critSect);
|
|
|
|
uint32_t 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;
|
|
uint32_t* tempSSRCVector = new uint32_t[newSize];
|
|
memcpy(tempSSRCVector, _ssrcVector, _sizeOfSSRC*sizeof(uint32_t));
|
|
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;
|
|
}
|
|
|
|
int32_t
|
|
SSRCDatabase::RegisterSSRC(const uint32_t ssrc)
|
|
{
|
|
CriticalSectionScoped lock(_critSect);
|
|
|
|
#ifndef WEBRTC_NO_STL
|
|
|
|
_ssrcMap[ssrc] = 0;
|
|
|
|
#else
|
|
if(_sizeOfSSRC <= _numberOfSSRC)
|
|
{
|
|
// allocate more space
|
|
const int newSize = _sizeOfSSRC + 10;
|
|
uint32_t* tempSSRCVector = new uint32_t[newSize];
|
|
memcpy(tempSSRCVector, _ssrcVector, _sizeOfSSRC*sizeof(uint32_t));
|
|
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;
|
|
}
|
|
|
|
int32_t
|
|
SSRCDatabase::ReturnSSRC(const uint32_t 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 uint32_t[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__);
|
|
}
|
|
|
|
uint32_t SSRCDatabase::GenerateRandom()
|
|
{
|
|
uint32_t ssrc = 0;
|
|
do
|
|
{
|
|
ssrc = rand();
|
|
ssrc = ssrc <<16;
|
|
ssrc += rand();
|
|
|
|
} while (ssrc == 0 || ssrc == 0xffffffff);
|
|
|
|
return ssrc;
|
|
}
|
|
} // namespace webrtc
|