2018-06-26 01:30:36 +00:00
|
|
|
#ifndef LLARP_CODEL_QUEUE_HPP
|
|
|
|
#define LLARP_CODEL_QUEUE_HPP
|
|
|
|
#include <llarp/time.h>
|
|
|
|
#include <cmath>
|
|
|
|
#include <functional>
|
|
|
|
#include <mutex>
|
|
|
|
#include <queue>
|
|
|
|
|
|
|
|
namespace llarp
|
|
|
|
{
|
|
|
|
namespace util
|
|
|
|
{
|
2018-06-26 17:12:33 +00:00
|
|
|
template < typename T, typename GetTime, llarp_time_t dropMs = 20,
|
|
|
|
llarp_time_t initialIntervalMs = 50 >
|
2018-06-26 01:30:36 +00:00
|
|
|
struct CoDelQueue
|
|
|
|
{
|
|
|
|
struct CoDelCompare
|
|
|
|
{
|
|
|
|
GetTime getTime = GetTime();
|
|
|
|
bool
|
|
|
|
operator()(const T& left, const T& right) const
|
|
|
|
{
|
|
|
|
return getTime(left) < getTime(right);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
void
|
|
|
|
Put(T* item)
|
|
|
|
{
|
|
|
|
std::unique_lock< std::mutex > lock(m_QueueMutex);
|
|
|
|
m_Queue.push(*item);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Process(std::queue< T >& result)
|
|
|
|
{
|
|
|
|
llarp_time_t lowest = 0xFFFFFFFFFFFFFFFFUL;
|
|
|
|
auto start = llarp_time_now_ms();
|
|
|
|
std::unique_lock< std::mutex > lock(m_QueueMutex);
|
|
|
|
while(m_Queue.size())
|
|
|
|
{
|
|
|
|
const auto& item = m_Queue.top();
|
|
|
|
auto dlt = start - getTime(item);
|
|
|
|
lowest = std::min(dlt, lowest);
|
|
|
|
if(m_Queue.size() == 1)
|
|
|
|
{
|
|
|
|
if(lowest > dropMs)
|
|
|
|
{
|
|
|
|
// drop
|
2018-06-26 13:39:29 +00:00
|
|
|
nextTickInterval += initialIntervalMs / std::sqrt(++dropNum);
|
|
|
|
llarp::Info("CoDel drop ", nextTickInterval, " ms next interval");
|
2018-06-26 01:30:36 +00:00
|
|
|
m_Queue.pop();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
nextTickInterval = initialIntervalMs;
|
|
|
|
dropNum = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
result.push(item);
|
|
|
|
m_Queue.pop();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
GetTime getTime = GetTime();
|
|
|
|
size_t dropNum = 0;
|
|
|
|
llarp_time_t nextTickInterval = initialIntervalMs;
|
|
|
|
std::mutex m_QueueMutex;
|
|
|
|
std::priority_queue< T, std::vector< T >, CoDelCompare > m_Queue;
|
|
|
|
};
|
|
|
|
} // namespace util
|
|
|
|
} // namespace llarp
|
|
|
|
|
|
|
|
#endif
|