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();
26utils::status blocks::IBlock::write_to_hardware() {
27 return utils::status(hardware->write_upscaling(scaling_factors) and hardware->write_outputs(outputs));
30bool blocks::IBlock::init() {
31 LOG(ANABRID_DEBUG_INIT, __PRETTY_FUNCTION__);
32 if (!FunctionBlock::init()) {
36 if (!write_to_hardware())
41bool blocks::IBlock::_is_connected(uint8_t input, uint8_t output)
const {
42 return outputs[output] & INPUT_BITMASK(input);
45bool blocks::IBlock::_is_output_connected(uint8_t output)
const {
return outputs[output]; }
47bool blocks::IBlock::is_connected(uint8_t input, uint8_t output)
const {
48 if (output >= NUM_OUTPUTS or input >= NUM_INPUTS)
50 return _is_connected(input, output);
53bool blocks::IBlock::is_anything_connected()
const {
54 for (
auto &output : OUTPUT_IDX_RANGE())
55 if (_is_output_connected(output))
60bool blocks::IBlock::connect(uint8_t input, uint8_t output,
bool exclusive,
61 bool allow_input_splitting) {
62 if (output >= NUM_OUTPUTS or input >= NUM_INPUTS)
66 if (!allow_input_splitting) {
67 for (
size_t other_output_idx = 0; other_output_idx < outputs.size(); other_output_idx++) {
68 if (output == other_output_idx)
70 if (is_connected(input, other_output_idx)) {
77 outputs[output] = INPUT_BITMASK(input);
79 outputs[output] |= INPUT_BITMASK(input);
83void blocks::IBlock::reset_outputs() {
84 for (
auto &output : outputs) {
89void blocks::IBlock::reset(entities::ResetAction action) {
90 FunctionBlock::reset(action);
92 if (action.has(entities::ResetAction::CIRCUIT_RESET)) {
102bool blocks::IBlock::disconnect(uint8_t input, uint8_t output) {
104 if (!is_connected(input, output))
106 outputs[output] &= ~INPUT_BITMASK(input);
110bool blocks::IBlock::disconnect(uint8_t output) {
111 if (output >= NUM_OUTPUTS)
117bool blocks::IBlock::set_upscaling(uint8_t input,
bool upscale) {
118 if (input >= NUM_INPUTS)
120 scaling_factors[input] = upscale;
124void blocks::IBlock::set_upscaling(std::bitset<NUM_INPUTS> scales) { scaling_factors = scales; }
126void blocks::IBlock::reset_upscaling() { scaling_factors.reset(); }
128bool blocks::IBlock::get_upscaling(uint8_t input)
const {
132 return scaling_factors[input];
135const std::array<uint32_t, blocks::IBlock::NUM_OUTPUTS> &blocks::IBlock::get_outputs()
const {
139void blocks::IBlock::set_outputs(
const std::array<uint32_t, NUM_OUTPUTS> &outputs_) {
144void blocks::IBlock::extract(entities::ExtractVisitor &collector) {
145 Entity::extract(collector);
146 auto& cfg = collector.create(pb_Config_sum_config_tag);
147 auto& sum_config = cfg.kind.sum_config = pb_SumConfig_init_default;
148 auto& sum_connection_idx = sum_config.connections_count;
149 for (
size_t out_idx = 0; out_idx < NUM_OUTPUTS; ++out_idx) {
150 auto &connection_config = sum_config.connections[sum_connection_idx];
151 connection_config.output = out_idx;
152 for (
size_t in_idx = 0; in_idx < NUM_INPUTS; ++in_idx) {
153 if (!_is_connected(in_idx, out_idx))
155 connection_config.inputs[connection_config.inputs_count++] = in_idx;
158 if (connection_config.inputs_count > 0)
159 ++sum_connection_idx;
162 for (
uint32_t in_idx = 0; in_idx < NUM_INPUTS; ++in_idx) {
163 sum_config.upscales[sum_config.upscales_count++] = {
165 .enabled = get_upscaling(in_idx)
170ConfigResult blocks::IBlock::config(
const pb_Config &cfg) {
171#ifdef ANABRID_DEBUG_ENTITY_CONFIG
172 Serial.println(__PRETTY_FUNCTION__);
174 if (cfg.which_kind != pb_Config_sum_config_tag) {
175 return ConfigResult::err(
"expected i block config");
178 auto& sum_coef = cfg.kind.sum_config;
179 auto res = apply_outputs(sum_coef);
183 auto res2 = apply_upscaling(sum_coef);
187 return ConfigResult::ok(
true);
190ConfigResult blocks::IBlock::apply_outputs(
const pb_SumConfig &cfg) {
194 reset(entities::ResetAction::CIRCUIT_RESET);
195 for (
size_t elem_idx = 0; elem_idx < cfg.connections_count; ++elem_idx) {
196 auto& connection = cfg.connections[elem_idx];
197 for (
size_t input_idx = 0; input_idx < connection.inputs_count; ++input_idx) {
198 auto input = connection.inputs[input_idx];
199 if (!connect(input, connection.output)) {
200 return ConfigResult::err_fmt(
"IBlock: Unable to connect input %d to output %d", input, connection.output);
205 return ConfigResult::ok(
true);
208ConfigResult blocks::IBlock::apply_upscaling(
const pb_SumConfig &cfg) {
209 for (
auto upscale : cfg.upscales) {
210 set_upscaling(upscale.lane, upscale.enabled);
212 return ConfigResult::ok(
true);