7#include <proto/main.pb.h>
9#include <block/iblock.h>
10#include <entity/visitor.h>
11#include <utils/is_number.h>
12#include <utils/logging.h>
14blocks::IBlock::IBlock(IBlockHAL *hardware) : FunctionBlock(
"I", hardware), hardware(hardware), outputs{0} {
15 classifier.class_enum = CLASS_;
18blocks::IBlock::IBlock() : IBlock(new IBlockHALDummy(
bus::NULL_ADDRESS)) {}
20metadata::eui_t blocks::IBlock::get_entity_eui()
const {
22 return hardware->get_entity_eui();
26UnitResult blocks::IBlock::write_to_hardware() {
27 if (!hardware->write_upscaling(scaling_factors))
28 return UnitResult::err(
"Writing i block scaling factors failed!");
30 if (!hardware->write_outputs(outputs))
31 return UnitResult::err(
"Writing i block outputs failed!");
33 return UnitResult::ok();
36UnitResult blocks::IBlock::init() {
37 LOG(ANABRID_DEBUG_INIT, __PRETTY_FUNCTION__);
38 TRY(FunctionBlock::init());
40 return write_to_hardware();
43bool blocks::IBlock::_is_connected(uint8_t input, uint8_t output)
const {
44 return outputs[output] & INPUT_BITMASK(input);
47bool blocks::IBlock::_is_output_connected(uint8_t output)
const {
return outputs[output]; }
49bool blocks::IBlock::is_connected(uint8_t input, uint8_t output)
const {
50 if (output >= NUM_OUTPUTS or input >= NUM_INPUTS)
52 return _is_connected(input, output);
55bool blocks::IBlock::is_anything_connected()
const {
56 for (
auto &output : OUTPUT_IDX_RANGE())
57 if (_is_output_connected(output))
62UnitResult blocks::IBlock::connect(uint8_t input, uint8_t output,
bool exclusive,
bool allow_input_splitting) {
63 if (output >= NUM_OUTPUTS)
64 return UnitResult::err_fmt(
"Error i-block connect output idx out of range: %d", output);
65 if (input >= NUM_INPUTS)
66 return UnitResult::err_fmt(
"Error i-block connect input idx out of range: %d", input);
69 if (!allow_input_splitting) {
70 for (
size_t other_output_idx = 0; other_output_idx < outputs.size(); other_output_idx++) {
71 if (output == other_output_idx)
73 if (is_connected(input, other_output_idx)) {
74 return UnitResult::err_fmt(
"Error i-block connect input is already used: %d", input);
80 outputs[output] = INPUT_BITMASK(input);
82 outputs[output] |= INPUT_BITMASK(input);
83 return UnitResult::ok();
86void blocks::IBlock::reset_outputs() {
87 for (
auto &output : outputs) {
92void blocks::IBlock::reset(entities::ResetAction action) {
93 FunctionBlock::reset(action);
95 if (action.has(entities::ResetAction::CIRCUIT_RESET)) {
105bool blocks::IBlock::disconnect(uint8_t input, uint8_t output) {
107 if (!is_connected(input, output))
109 outputs[output] &= ~INPUT_BITMASK(input);
113bool blocks::IBlock::disconnect(uint8_t output) {
114 if (output >= NUM_OUTPUTS)
120bool blocks::IBlock::set_upscaling(uint8_t input,
bool upscale) {
121 if (input >= NUM_INPUTS)
123 scaling_factors[input] = upscale;
127void blocks::IBlock::set_upscaling(std::bitset<NUM_INPUTS> scales) { scaling_factors = scales; }
129void blocks::IBlock::reset_upscaling() { scaling_factors.reset(); }
131bool blocks::IBlock::get_upscaling(uint8_t input)
const {
135 return scaling_factors[input];
138const std::array<uint32_t, blocks::IBlock::NUM_OUTPUTS> &blocks::IBlock::get_outputs()
const {
142void blocks::IBlock::set_outputs(
const std::array<uint32_t, NUM_OUTPUTS> &outputs_) {
147void blocks::IBlock::extract(entities::ExtractVisitor &collector) {
148 Entity::extract(collector);
150 if (!collector.include_configuration())
return;
151 auto& item = collector.create(pb_Item_sum_config_tag);
152 auto& sum_config = item.kind.sum_config = pb_SumConfig_init_default;
153 auto& sum_connection_idx = sum_config.connections_count;
154 for (
size_t out_idx = 0; out_idx < NUM_OUTPUTS; ++out_idx) {
155 auto &connection_config = sum_config.connections[sum_connection_idx];
156 connection_config.output = out_idx;
157 for (
size_t in_idx = 0; in_idx < NUM_INPUTS; ++in_idx) {
158 if (!_is_connected(in_idx, out_idx))
160 connection_config.inputs[connection_config.inputs_count++] = in_idx;
163 if (connection_config.inputs_count > 0)
164 ++sum_connection_idx;
167 for (
uint32_t in_idx = 0; in_idx < NUM_INPUTS; ++in_idx) {
168 sum_config.upscales[sum_config.upscales_count++] = {
169 .lane =
static_cast<uint8_t
>(in_idx),
170 .enabled = get_upscaling(in_idx)
175ConfigResult blocks::IBlock::config(
const pb_Item &item) {
176#ifdef ANABRID_DEBUG_ENTITY_CONFIG
177 Serial.println(__PRETTY_FUNCTION__);
179 if (item.which_kind != pb_Item_sum_config_tag) {
180 return ConfigResult::err(
"expected i block config");
183 auto& sum_coef = item.kind.sum_config;
184 auto res = apply_outputs(sum_coef);
188 auto res2 = apply_upscaling(sum_coef);
192 return ConfigResult::ok(
true);
195ConfigResult blocks::IBlock::apply_outputs(
const pb_SumConfig &item) {
199 reset(entities::ResetAction::CIRCUIT_RESET);
200 for (
size_t elem_idx = 0; elem_idx < item.connections_count; ++elem_idx) {
201 auto& connection = item.connections[elem_idx];
202 for (
size_t input_idx = 0; input_idx < connection.inputs_count; ++input_idx) {
203 auto input = connection.inputs[input_idx];
204 if (!connect(input, connection.output)) {
205 return ConfigResult::err_fmt(
"IBlock: Unable to connect input %d to output %d", input, connection.output);
210 return ConfigResult::ok(
true);
213ConfigResult blocks::IBlock::apply_upscaling(
const pb_SumConfig &item) {
214 for (
auto upscale : item.upscales) {
215 set_upscaling(upscale.lane, upscale.enabled);
217 return ConfigResult::ok(
true);