libuv — ограничение скорости обратного вызова для бездействующего события без блокировки потока без многопоточности

Я использую libsourcey, который использует libuv в качестве базового сетевого уровня ввода / вывода.
Все настроено и, кажется, работает (еще ничего не тестировал, так как я только создаю прототипы и экспериментирую). Тем не менее, я требую, чтобы рядом с циклом приложения (тот, который поставляется с libsourcey, который опирается на цикл libuv), также вызывается «Idle function». Как и сейчас, он вызывает Idle CB на каждом цикле, который сильно загружает процессор. Мне нужен способ ограничить скорость вызова uv_idle_cb, не блокируя вызывающий поток, который является тем же самым, что приложение использует для обработки данных ввода / вывода (не уверен насчет этого последнего утверждения, поправьте меня, если я ошибаюсь).

Функция бездействия будет управлять несколькими различными аспектами приложения, и она должна запускаться только x раз в течение 1 секунды. Кроме того, все должно работать в одном и том же потоке (планируется обновить сетевую инфраструктуру старого приложения, которая работает полностью однопоточным).

Это код, который у меня есть до сих пор, который также включает в себя тест, который я провел со спящим потоком в обратном вызове, но он блокирует все, так что даже второй установленный мной неактивный cb имеет такую ​​же скорость вызова, как и первый.

struct TCPServers
{
CTCPManager<scy::net::SSLSocket> ssl;
};

int counter = 0;
void idle_cb(uv_idle_t *handle)
{
printf("Idle callback %d TID %d\n", counter, std::this_thread::get_id());

counter++;

std::this_thread::sleep_for(std::chrono::milliseconds(1000 / 25));
}

int counter2 = 0;
void idle_cb2(uv_idle_t *handle)
{
printf("Idle callback2 %d TID %d\n", counter2, std::this_thread::get_id());

counter2++;

std::this_thread::sleep_for(std::chrono::milliseconds(1000 / 50));
}

class CApplication : public scy::Application
{
public:
CApplication() : scy::Application(), m_uvIdleCallback(nullptr), m_bUseSSL(false)
{}

void start()
{
run();

if (m_uvIdleCallback)
uv_idle_start(&m_uvIdle, m_uvIdleCallback);

if (m_uvIdleCallback2)
uv_idle_start(&m_uvIdle2, m_uvIdleCallback2);
}

void stop()
{
scy::Application::stop();

uv_idle_stop(&m_uvIdle);

if (m_bUseSSL)
scy::net::SSLManager::instance().shutdown();
}

void bindIdleEvent(uv_idle_cb cb)
{
m_uvIdleCallback = cb;
uv_idle_init(loop, &m_uvIdle);
}

void bindIdleEvent2(uv_idle_cb cb)
{
m_uvIdleCallback2 = cb;
uv_idle_init(loop, &m_uvIdle2);
}

void initSSL(const std::string& privateKeyFile = "", const std::string& certificateFile = "")
{
scy::net::SSLManager::instance().initNoVerifyServer(privateKeyFile, certificateFile);
m_bUseSSL = true;
}

private:
uv_idle_t m_uvIdle;
uv_idle_t m_uvIdle2;
uv_idle_cb m_uvIdleCallback;
uv_idle_cb m_uvIdleCallback2;
bool m_bUseSSL;
};

int main()
{
CApplication app;
app.bindIdleEvent(idle_cb);
app.bindIdleEvent2(idle_cb2);
app.initSSL();
app.start();

TCPServers srvs;
srvs.ssl.start("127.0.0.1", 9000);

app.waitForShutdown([&](void*) {
srvs.ssl.shutdown();
});

app.stop();

system("PAUSE");
return 0;
}

Спасибо заранее, если кто-нибудь может помочь.

0

Решение

Решил проблему, используя uv_timer_t и uv_timer_cb (еще не копался в документации libuv). Загрузка ЦП резко снизилась, и ничто не блокируется.

0

Другие решения

Других решений пока нет …

По вопросам рекламы [email protected]