187 lines
3.9 KiB
C++
187 lines
3.9 KiB
C++
#include "api/processor/processor.h"
|
|
|
|
#include <glog/logging.h>
|
|
|
|
#include <utility>
|
|
|
|
MYNTEYE_BEGIN_NAMESPACE
|
|
|
|
Processor::Processor()
|
|
: activated_(false),
|
|
input_ready_(false),
|
|
idle_(true),
|
|
dropped_count_(0),
|
|
input_(nullptr),
|
|
output_(nullptr),
|
|
output_result_(nullptr),
|
|
pre_callback_(nullptr),
|
|
post_callback_(nullptr),
|
|
callback_(nullptr),
|
|
parent_(nullptr) {
|
|
VLOG(2) << __func__;
|
|
}
|
|
|
|
Processor::~Processor() {
|
|
VLOG(2) << __func__;
|
|
Deactivate();
|
|
input_.reset(nullptr);
|
|
output_.reset(nullptr);
|
|
output_result_.reset(nullptr);
|
|
childs_.clear();
|
|
}
|
|
|
|
std::string Processor::Name() {
|
|
return "Processor";
|
|
}
|
|
|
|
void Processor::AddChild(const std::shared_ptr<Processor> &child) {
|
|
child->parent_ = this;
|
|
childs_.push_back(child);
|
|
}
|
|
|
|
void Processor::RemoveChild(const std::shared_ptr<Processor> &child) {
|
|
childs_.remove(child);
|
|
}
|
|
|
|
std::list<std::shared_ptr<Processor>> Processor::GetChilds() {
|
|
return childs_;
|
|
}
|
|
|
|
void Processor::SetPreProcessCallback(PreProcessCallback callback) {
|
|
pre_callback_ = std::move(callback);
|
|
}
|
|
|
|
void Processor::SetPostProcessCallback(PostProcessCallback callback) {
|
|
post_callback_ = std::move(callback);
|
|
}
|
|
|
|
void Processor::SetProcessCallback(ProcessCallback callback) {
|
|
callback_ = std::move(callback);
|
|
}
|
|
|
|
void Processor::Activate(bool tree) {
|
|
if (activated_)
|
|
return;
|
|
if (tree) {
|
|
// Activate all parents
|
|
Processor *parent = parent_;
|
|
while (parent != nullptr) {
|
|
parent->Activate();
|
|
parent = parent->parent_;
|
|
}
|
|
}
|
|
activated_ = true;
|
|
thread_ = std::thread(&Processor::Run, this);
|
|
// thread_.detach();
|
|
}
|
|
|
|
void Processor::Deactivate(bool tree) {
|
|
if (!activated_)
|
|
return;
|
|
if (tree) {
|
|
// Deactivate all childs
|
|
iterate_processors(GetChilds(), [](std::shared_ptr<Processor> proc) {
|
|
proc->Deactivate();
|
|
});
|
|
}
|
|
activated_ = false;
|
|
{
|
|
std::lock_guard<std::mutex> lk(mtx_input_ready_);
|
|
input_ready_ = true;
|
|
}
|
|
cond_input_ready_.notify_all();
|
|
thread_.join();
|
|
}
|
|
|
|
bool Processor::IsActivated() {
|
|
return activated_;
|
|
}
|
|
|
|
bool Processor::IsIdle() {
|
|
std::lock_guard<std::mutex> lk(mtx_state_);
|
|
return idle_;
|
|
}
|
|
|
|
bool Processor::Process(const Object *const in) {
|
|
if (!activated_)
|
|
return false;
|
|
if (!idle_) {
|
|
std::lock_guard<std::mutex> lk(mtx_state_);
|
|
if (!idle_) {
|
|
++dropped_count_;
|
|
return false;
|
|
}
|
|
}
|
|
{
|
|
std::lock_guard<std::mutex> lk(mtx_input_ready_);
|
|
input_.reset(in->Clone());
|
|
input_ready_ = true;
|
|
}
|
|
cond_input_ready_.notify_all();
|
|
return true;
|
|
}
|
|
|
|
Object *Processor::GetOutput() {
|
|
std::lock_guard<std::mutex> lk(mtx_result_);
|
|
return output_result_.get();
|
|
}
|
|
|
|
std::uint64_t Processor::GetDroppedCount() {
|
|
std::lock_guard<std::mutex> lk(mtx_state_);
|
|
return dropped_count_;
|
|
}
|
|
|
|
void Processor::Run() {
|
|
VLOG(2) << Name() << " thread start";
|
|
while (true) {
|
|
std::unique_lock<std::mutex> lk(mtx_input_ready_);
|
|
cond_input_ready_.wait(lk, [this] { return input_ready_; });
|
|
|
|
if (!activated_) {
|
|
SetIdle(true);
|
|
input_ready_ = false;
|
|
break;
|
|
}
|
|
SetIdle(false);
|
|
|
|
if (!output_) {
|
|
output_.reset(OnCreateOutput());
|
|
}
|
|
|
|
if (pre_callback_) {
|
|
pre_callback_(input_.get());
|
|
}
|
|
if (callback_) {
|
|
if (!callback_(input_.get(), output_.get(), parent_)) {
|
|
OnProcess(input_.get(), output_.get(), parent_);
|
|
}
|
|
} else {
|
|
OnProcess(input_.get(), output_.get(), parent_);
|
|
}
|
|
if (post_callback_) {
|
|
post_callback_(output_.get());
|
|
}
|
|
{
|
|
std::unique_lock<std::mutex> lk(mtx_result_);
|
|
output_result_.reset(output_->Clone());
|
|
}
|
|
|
|
if (!childs_.empty()) {
|
|
for (auto child : childs_) {
|
|
child->Process(output_.get());
|
|
}
|
|
}
|
|
|
|
SetIdle(true);
|
|
input_ready_ = false;
|
|
}
|
|
VLOG(2) << Name() << " thread end";
|
|
}
|
|
|
|
void Processor::SetIdle(bool idle) {
|
|
std::lock_guard<std::mutex> lk(mtx_state_);
|
|
idle_ = idle;
|
|
}
|
|
|
|
MYNTEYE_END_NAMESPACE
|