TP1
This commit is contained in:
153
src/PoolRouter.cpp
Normal file
153
src/PoolRouter.cpp
Normal file
@@ -0,0 +1,153 @@
|
||||
/**
|
||||
* @file PoolRouter.cpp
|
||||
* @author The ARTIS Development Team
|
||||
* See the AUTHORS or Authors.txt file
|
||||
*/
|
||||
|
||||
/*
|
||||
* ARTIS - the multimodeling and simulation environment
|
||||
* This file is a part of the ARTIS environment
|
||||
*
|
||||
* Copyright (C) 2013-2023 ULCO http://www.univ-littoral.fr
|
||||
*
|
||||
* 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, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#include "PoolRouter.hpp"
|
||||
|
||||
namespace artis::factory {
|
||||
|
||||
void PoolRouter::dint(const Time & /* t */) {
|
||||
switch (_phase) {
|
||||
case Phase::INIT: {
|
||||
_phase = Phase::WAIT;
|
||||
break;
|
||||
}
|
||||
case Phase::WAIT: {
|
||||
break;
|
||||
}
|
||||
case Phase::SEND: {
|
||||
auto it = next_po();
|
||||
|
||||
assert(it != _pending_po.cend());
|
||||
|
||||
if ((*it)->is_finish()) {
|
||||
_available_machines[(*it)->last_operation()->get_loc_index()] = true;
|
||||
} else if ((*it)->current_operation().get_type() == CHANGE_POOL_TYPE) {
|
||||
_available_machines[(*it)->previous_operation()->get_loc_index()] = true;
|
||||
} else {
|
||||
_available_machines[(*it)->current_operation().get_loc_index()] = false;
|
||||
}
|
||||
_pending_po.erase(it);
|
||||
if (std::find_if(_pending_po.cbegin(), _pending_po.cend(), [this](const auto &e) {
|
||||
return e->is_finish() or e->current_operation().get_type() == CHANGE_POOL_TYPE or
|
||||
_available_machines[e->current_operation().get_loc_index()];
|
||||
}) != _pending_po.cend()) {
|
||||
_phase = Phase::SEND;
|
||||
} else {
|
||||
_phase = Phase::WAIT;
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void PoolRouter::dext(const Time &t, const Time & /* e */, const Bag &bag) {
|
||||
std::for_each(bag.begin(), bag.end(), [t, this](const common::event::ExternalEvent<common::DoubleTime> &event) {
|
||||
if (event.port_index() == inputs::IN) {
|
||||
uint8_t *data = nullptr;
|
||||
|
||||
event.data()(data);
|
||||
_pending_po.push_back(std::make_unique<ProductionOrder>(data, event.data().size()));
|
||||
_pending_po.back()->next();
|
||||
} else if (event.port_index() >= inputs::IN_M) {
|
||||
uint8_t *data = nullptr;
|
||||
|
||||
event.data()(data);
|
||||
_pending_po.push_back(std::make_unique<ProductionOrder>(data, event.data().size()));
|
||||
_available_machines[event.port_index() - inputs::IN_M] = true;
|
||||
_pending_po.back()->next();
|
||||
}
|
||||
}
|
||||
);
|
||||
if (std::find_if(_pending_po.cbegin(), _pending_po.cend(), [this](const auto &e) {
|
||||
return e->is_finish() or e->current_operation().get_type() == CHANGE_POOL_TYPE or
|
||||
_available_machines[e->current_operation().get_loc_index()];
|
||||
}) != _pending_po.cend()) {
|
||||
_phase = Phase::SEND;
|
||||
} else {
|
||||
_phase = Phase::WAIT;
|
||||
}
|
||||
}
|
||||
|
||||
void PoolRouter::start(const Time & /* t */) {
|
||||
_phase = Phase::INIT;
|
||||
}
|
||||
|
||||
Time PoolRouter::ta(const Time & /* t */) const {
|
||||
switch (_phase) {
|
||||
case Phase::INIT:
|
||||
return 0;
|
||||
case Phase::WAIT:
|
||||
return artis::common::DoubleTime::infinity;
|
||||
case Phase::SEND:
|
||||
return 0;
|
||||
}
|
||||
return artis::common::DoubleTime::infinity;
|
||||
}
|
||||
|
||||
Bag PoolRouter::lambda(const Time & /* t */) const {
|
||||
Bag bag;
|
||||
|
||||
if (_phase == Phase::SEND) {
|
||||
if (not _pending_po.empty()) {
|
||||
auto it = next_po();
|
||||
|
||||
assert(it != _pending_po.cend());
|
||||
|
||||
auto po = it->get();
|
||||
|
||||
if (po->is_finish() or po->current_operation().get_type() == CHANGE_POOL_TYPE) {
|
||||
bag.push_back(ExternalEvent(outputs::OUT, common::event::Value(po->_buffer, po->_size)));
|
||||
} else if (po->current_operation().get_type() == CHANGE_MACHINE_TYPE) {
|
||||
bag.push_back(ExternalEvent(outputs::OUT_M + po->current_operation().get_loc_index(),
|
||||
common::event::Value(po->_buffer, po->_size)));
|
||||
}
|
||||
}
|
||||
}
|
||||
return bag;
|
||||
}
|
||||
|
||||
artis::common::event::Value PoolRouter::observe(const Time & /* t */, unsigned int index) const {
|
||||
switch (index) {
|
||||
case vars::WAITING_PO_NUMBER:
|
||||
return (unsigned int) _pending_po.size();
|
||||
}
|
||||
return {};
|
||||
}
|
||||
|
||||
PoolRouter::ProductionOrders::const_iterator PoolRouter::next_po() const {
|
||||
auto it = std::find_if(_pending_po.cbegin(), _pending_po.cend(), [](const auto &e) {
|
||||
return e->is_finish() or e->current_operation().get_type() == CHANGE_POOL_TYPE;
|
||||
});
|
||||
|
||||
if (it != _pending_po.cend()) {
|
||||
return it;
|
||||
} else {
|
||||
return std::find_if(_pending_po.cbegin(), _pending_po.cend(),
|
||||
[this](const auto &e) { return _available_machines[e->current_operation().get_loc_index()]; });
|
||||
}
|
||||
}
|
||||
|
||||
} // namespace artis::factory
|
||||
Reference in New Issue
Block a user