blob: 9d0ebb1cfa37fd8115523f0590e4c4c41eb59832 [file] [log] [blame]
/*
* Copyright (C) 2004-2021 Savoir-faire Linux Inc.
*
* Author: Guillaume Roguez <Guillaume.Roguez@savoirfairelinux.com>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/
#include "threadloop.h"
#include "logger.h"
#include <ciso646> // fix windows compiler bug
namespace jami {
void
ThreadLoop::mainloop(std::thread::id& tid,
const std::function<bool()> setup,
const std::function<void()> process,
const std::function<void()> cleanup)
{
tid = std::this_thread::get_id();
try {
if (setup()) {
while (state_ == ThreadState::RUNNING)
process();
cleanup();
} else {
JAMI_ERR("setup failed");
}
} catch (const ThreadLoopException& e) {
JAMI_ERR("[threadloop:%p] ThreadLoopException: %s", this, e.what());
} catch (const std::exception& e) {
JAMI_ERR("[threadloop:%p] Unwaited exception: %s", this, e.what());
}
stop();
}
ThreadLoop::ThreadLoop(const std::function<bool()>& setup,
const std::function<void()>& process,
const std::function<void()>& cleanup)
: setup_(setup)
, process_(process)
, cleanup_(cleanup)
, thread_()
{}
ThreadLoop::~ThreadLoop()
{
if (isRunning()) {
JAMI_ERR("join() should be explicitly called in owner's destructor");
join();
}
}
void
ThreadLoop::start()
{
const auto s = state_.load();
if (s == ThreadState::RUNNING) {
JAMI_ERR("already started");
return;
}
// stop pending but not processed by thread yet?
if (s == ThreadState::STOPPING and thread_.joinable()) {
JAMI_DBG("stop pending");
thread_.join();
}
state_ = ThreadState::RUNNING;
thread_ = std::thread(&ThreadLoop::mainloop, this, std::ref(threadId_), setup_, process_, cleanup_);
threadId_ = thread_.get_id();
}
void
ThreadLoop::stop()
{
if (state_ == ThreadState::RUNNING)
state_ = ThreadState::STOPPING;
}
void
ThreadLoop::join()
{
stop();
if (thread_.joinable())
thread_.join();
}
void
ThreadLoop::waitForCompletion()
{
if (thread_.joinable())
thread_.join();
}
void
ThreadLoop::exit()
{
stop();
throw ThreadLoopException();
}
bool
ThreadLoop::isRunning() const noexcept
{
#ifdef _WIN32
return state_ == ThreadState::RUNNING;
#else
return thread_.joinable() and state_ == ThreadState::RUNNING;
#endif
}
void
InterruptedThreadLoop::stop()
{
ThreadLoop::stop();
cv_.notify_one();
}
} // namespace jami