Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Apply rate limiting to Pan as well as Gain #3429

Merged
merged 2 commits into from
Dec 6, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
96 changes: 65 additions & 31 deletions src/client.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -176,9 +176,9 @@ CClient::CClient ( const quint16 iPortNumber,
PreciseTime.start();

// set gain delay timer to single-shot and connect handler function
TimerGain.setSingleShot ( true );
TimerGainOrPan.setSingleShot ( true );

QObject::connect ( &TimerGain, &QTimer::timeout, this, &CClient::OnTimerRemoteChanGain );
QObject::connect ( &TimerGainOrPan, &QTimer::timeout, this, &CClient::OnTimerRemoteChanGainOrPan );

// start the socket (it is important to start the socket after all
// initializations and connections)
Expand Down Expand Up @@ -431,27 +431,28 @@ void CClient::SetDoAutoSockBufSize ( const bool bValue )
CreateServerJitterBufferMessage();
}

// In order not to flood the server with gain change messages, particularly when using
// In order not to flood the server with gain or pan change messages, particularly when using
// a MIDI controller, a timer is used to limit the rate at which such messages are generated.
// This avoids a potential long backlog of messages, since each must be ACKed before the next
// can be sent, and this ACK is subject to the latency of the server connection.
//
// When the first gain change message is requested after an idle period (i.e. the timer is not
// running), it will be sent immediately, and a 300ms timer started.
// When the first gain or pan change message is requested after an idle period (i.e. the timer is not
// running), it will be sent immediately, and a timer started. The timer period is dependent on
// the current ping time to the remote server.
//
// If a gain change message is requested while the timer is still running, the new gain is not sent,
// but just stored in clientChannels[iId].newGain, and the minGainId and maxGainId updated to note
// the range of IDs that must be checked when the time expires (this will usually be a single channel
// unless channel grouping is being used). This avoids having to check all possible channels.
// If a gain or pan change message is requested while the timer is still running, the new value is not sent,
// but just stored in newGain or newPan within clientChannels[iId], and the minGainOrPanId and maxGainOrPanId
// updated to note the range of IDs that must be checked when the time expires (this will usually be a single
// channel unless channel grouping is being used). This avoids having to check all possible channels.
//
// When the timer fires, the channels minGainId <= iId < maxGainId are checked by comparing the last sent value
// in clientChannels[iId].oldGain with any pending value in clientChannels[iId].newGain, and if they differ,
// the new value is sent, updating clientChannels[iId].oldGain with the sent value. If any new values are
// sent, the timer is restarted so that further immediate updates will be pended.
// When the timer fires, the channels minGainOrPanId <= iId < maxGainOrPanId are checked by comparing the
// last sent values in oldGain or oldPan with any pending values in newGain or newPan, and if they differ,
// the new value is sent, updating oldGain or oldPan with the sent value. If any new values are sent,
// the timer is restarted so that further immediate updates will be pended.

void CClient::SetRemoteChanGain ( const int iId, const float fGain, const bool bIsMyOwnFader )
{
QMutexLocker locker ( &MutexGain );
QMutexLocker locker ( &MutexGainOrPan );

CClientChannel* clientChan = &clientChannels[iId];

Expand All @@ -461,17 +462,17 @@ void CClient::SetRemoteChanGain ( const int iId, const float fGain, const bool b
fMuteOutStreamGain = fGain;
}

if ( TimerGain.isActive() )
if ( TimerGainOrPan.isActive() )
{
// just update the new value for sending later;
// will compare with oldGain when the timer fires
clientChan->newGain = fGain;

// update range of channel IDs to check in the timer
if ( iId < minGainId )
minGainId = iId; // first value to check
if ( iId >= maxGainId )
maxGainId = iId + 1; // first value NOT to check
if ( iId < minGainOrPanId )
minGainOrPanId = iId; // first value to check
if ( iId >= maxGainOrPanId )
maxGainOrPanId = iId + 1; // first value NOT to check

return;
}
Expand All @@ -481,15 +482,15 @@ void CClient::SetRemoteChanGain ( const int iId, const float fGain, const bool b
clientChan->oldGain = clientChan->newGain = fGain;
Channel.SetRemoteChanGain ( clientChan->iServerChannelID, fGain ); // translate client channel to server channel ID

StartDelayTimer();
StartTimerGainOrPan();
}

void CClient::OnTimerRemoteChanGain()
void CClient::OnTimerRemoteChanGainOrPan()
{
QMutexLocker locker ( &MutexGain );
QMutexLocker locker ( &MutexGainOrPan );
bool bSent = false;

for ( int iId = minGainId; iId < maxGainId; iId++ )
for ( int iId = minGainOrPanId; iId < maxGainOrPanId; iId++ )
{
CClientChannel* clientChan = &clientChannels[iId];

Expand All @@ -500,37 +501,69 @@ void CClient::OnTimerRemoteChanGain()
Channel.SetRemoteChanGain ( clientChan->iServerChannelID, fGain ); // translate client channel to server channel ID
bSent = true;
}

if ( clientChan->newPan != clientChan->oldPan )
{
// send new pan and record as old pan
float fPan = clientChan->oldPan = clientChan->newPan;
Channel.SetRemoteChanPan ( clientChan->iServerChannelID, fPan ); // translate client channel to server channel ID
bSent = true;
}
}

// if a new gain has been sent, reset the range of channel IDs to empty and start timer
// if a new gain or pan has been sent, reset the range of channel IDs to empty and start timer
pljones marked this conversation as resolved.
Show resolved Hide resolved
if ( bSent )
{
StartDelayTimer();
StartTimerGainOrPan();
}
}

// reset the range of channel IDs to check and start the delay timer
void CClient::StartDelayTimer()
void CClient::StartTimerGainOrPan()
{
maxGainId = 0;
minGainId = MAX_NUM_CHANNELS;
maxGainOrPanId = 0;
minGainOrPanId = MAX_NUM_CHANNELS;

// start timer to delay sending further updates
// use longer delay when connected to server with higher ping time,
// double the ping time in order to allow a bit of overhead for other messages
if ( iCurPingTime < DEFAULT_GAIN_DELAY_PERIOD_MS / 2 )
{
TimerGain.start ( DEFAULT_GAIN_DELAY_PERIOD_MS );
TimerGainOrPan.start ( DEFAULT_GAIN_DELAY_PERIOD_MS );
}
else
{
TimerGain.start ( iCurPingTime * 2 );
TimerGainOrPan.start ( iCurPingTime * 2 );
}
}

void CClient::SetRemoteChanPan ( const int iId, const float fPan )
{
Channel.SetRemoteChanPan ( clientChannels[iId].iServerChannelID, fPan ); // translate client channel to server channel ID
QMutexLocker locker ( &MutexGainOrPan );

CClientChannel* clientChan = &clientChannels[iId];

if ( TimerGainOrPan.isActive() )
{
// just update the new value for sending later;
// will compare with oldPan when the timer fires
clientChan->newPan = fPan;

// update range of channel IDs to check in the timer
if ( iId < minGainOrPanId )
minGainOrPanId = iId; // first value to check
if ( iId >= maxGainOrPanId )
maxGainOrPanId = iId + 1; // first value NOT to check

return;
}

// here the timer was not active:
// send the actual gain and reset the range of channel IDs to empty
clientChan->oldPan = clientChan->newPan = fPan;
Channel.SetRemoteChanPan ( clientChan->iServerChannelID, fPan ); // translate client channel to server channel ID

StartTimerGainOrPan();
}

bool CClient::SetServerAddr ( QString strNAddr )
Expand Down Expand Up @@ -1541,6 +1574,7 @@ int CClient::FindClientChannel ( const int iServerChannelID, const bool bCreateI
clientChan->iJoinSequence = ++iJoinSequence;

clientChan->oldGain = clientChan->newGain = 1.0f;
clientChan->oldPan = clientChan->newPan = 0.5f;

clientChan->level = 0;

Expand Down
18 changes: 8 additions & 10 deletions src/client.h
Original file line number Diff line number Diff line change
Expand Up @@ -112,6 +112,7 @@ class CClientChannel
int iJoinSequence; // order of joining of session participants

float oldGain, newGain; // for rate-limiting sending of gain messages
float oldPan, newPan; // for rate-limiting sending of pan messages

uint16_t level; // last value of level meter received for channel

Expand Down Expand Up @@ -255,10 +256,9 @@ class CClient : public QObject
void SetMuteOutStream ( const bool bDoMute ) { bMuteOutStream = bDoMute; }

void SetRemoteChanGain ( const int iId, const float fGain, const bool bIsMyOwnFader );
void OnTimerRemoteChanGain();
void StartDelayTimer();

void SetRemoteChanPan ( const int iId, const float fPan );
void OnTimerRemoteChanGainOrPan();
softins marked this conversation as resolved.
Show resolved Hide resolved
void StartTimerGainOrPan();

void SetInputBoost ( const int iNewBoost ) { iInputBoost = iNewBoost; }

Expand Down Expand Up @@ -393,13 +393,11 @@ class CClient : public QObject
// for ping measurement
QElapsedTimer PreciseTime;

// for gain rate limiting
QMutex MutexGain;
QTimer TimerGain;
int minGainId;
int maxGainId;
float oldGain[MAX_NUM_CHANNELS];
float newGain[MAX_NUM_CHANNELS];
// for gain or pan rate limiting
QMutex MutexGainOrPan;
QTimer TimerGainOrPan;
int minGainOrPanId;
int maxGainOrPanId;
int iCurPingTime;

CSignalHandler* pSignalHandler;
Expand Down
Loading