5#include <proto/main.pb.h>
9#include <block/ublock.h>
10#include <entity/visitor.h>
11#include <utils/logging.h>
13void utils::shift_5_left(uint8_t *buffer,
size_t size) {
14 for (
size_t idx = 0; idx <
size - 1; idx++) {
15 auto next = buffer[idx + 1];
18 buffer[idx] |= (next >> 3) & 0x1F;
20 buffer[
size - 1] <<= 5;
23blocks::UBlock::UBlock(UBlockHAL *hardware)
24 : FunctionBlock(
"U", hardware), hardware(hardware), output_input_map{} {
25 classifier.class_enum = CLASS_;
29metadata::eui_t blocks::UBlock::get_entity_eui()
const {
31 return hardware->get_entity_eui();
35bool blocks::UBlock::_i_sanity_check(
const uint8_t input) {
return input < NUM_OF_INPUTS; }
37bool blocks::UBlock::_o_sanity_check(
const uint8_t output) {
return output < NUM_OF_OUTPUTS; }
39bool blocks::UBlock::_io_sanity_check(
const uint8_t input,
const uint8_t output) {
40 return _i_sanity_check(input) && _o_sanity_check(output);
43bool blocks::UBlock::connect(
const uint8_t input,
const uint8_t output,
bool force) {
44 if (!_io_sanity_check(input, output))
47 output_input_map[output] = input;
51bool blocks::UBlock::connect_alternative(Transmission_Mode signal_mode,
const uint8_t output,
52 bool force,
bool use_a_side) {
53 if (!_o_sanity_check(output))
57 if (signal_mode == Transmission_Mode::ANALOG_INPUT)
61 if (!force and _is_output_connected(output))
66 if (a_side_mode != signal_mode && use_a_side) {
71 change_a_side_transmission_mode(signal_mode);
74 if (b_side_mode != signal_mode && !use_a_side) {
76 if (is_input_connected(14) || is_input_connected(15))
78 change_b_side_transmission_mode(signal_mode);
87 else if (output == 15)
89 else if (output == 30)
100 connect(input, output);
104void blocks::UBlock::_disconnect(
const uint8_t output) { output_input_map[output] = -1; }
106bool blocks::UBlock::disconnect(
const uint8_t input,
const uint8_t output) {
107 if (!_io_sanity_check(input, output))
110 if (_is_connected(input, output)) {
118bool blocks::UBlock::disconnect(
const uint8_t output) {
119 if (!_o_sanity_check(output))
125bool blocks::UBlock::_is_connected(
const uint8_t input,
const uint8_t output)
const {
126 return output_input_map[output] == input;
129bool blocks::UBlock::is_connected(
const uint8_t input,
const uint8_t output)
const {
130 if (!_io_sanity_check(input, output))
133 return _is_connected(input, output);
136bool blocks::UBlock::_is_output_connected(
const uint8_t output)
const {
137 return output_input_map[output] >= 0;
140bool blocks::UBlock::is_output_connected(
const uint8_t output)
const {
141 if (!_o_sanity_check(output))
144 return _is_output_connected(output);
147bool blocks::UBlock::_is_input_connected(
const uint8_t input)
const {
148 for (
const auto &output : output_input_map)
154bool blocks::UBlock::is_input_connected(
const uint8_t input)
const {
155 if (!_i_sanity_check(input))
157 return _is_input_connected(input);
160bool blocks::UBlock::is_anything_connected()
const {
161 for (
auto output : OUTPUT_IDX_RANGE())
162 if (_is_output_connected(output))
167void blocks::UBlock::change_a_side_transmission_mode(
const Transmission_Mode
mode) {
171void blocks::UBlock::change_b_side_transmission_mode(
const Transmission_Mode
mode) {
175void blocks::UBlock::change_all_transmission_modes(
const Transmission_Mode
mode) {
176 change_a_side_transmission_mode(
mode);
177 change_b_side_transmission_mode(
mode);
181blocks::UBlock::change_all_transmission_modes(
const std::pair<Transmission_Mode, Transmission_Mode> modes) {
182 change_a_side_transmission_mode(modes.first);
183 change_b_side_transmission_mode(modes.second);
186std::pair<blocks::UBlock::Transmission_Mode, blocks::UBlock::Transmission_Mode>
187blocks::UBlock::get_all_transmission_modes()
const {
188 return std::make_pair(a_side_mode, b_side_mode);
191blocks::UBlock::Reference_Magnitude blocks::UBlock::get_reference_magnitude() {
192 return ref_magnitude;
195void blocks::UBlock::change_reference_magnitude(blocks::UBlock::Reference_Magnitude ref) {
199utils::status blocks::UBlock::write_to_hardware() {
200 if (!hardware->write_outputs(output_input_map) or
201 !hardware->write_transmission_modes_and_ref({a_side_mode, b_side_mode}, ref_magnitude)) {
202 LOG(ANABRID_PEDANTIC, __PRETTY_FUNCTION__);
203 return utils::status::failure();
205 return utils::status::success();
208void blocks::UBlock::reset_connections() {
209 std::fill(begin(output_input_map), end(output_input_map), -1);
212void blocks::UBlock::reset(entities::ResetAction action) {
213 FunctionBlock::reset(action);
215 if (action.has(entities::ResetAction::CIRCUIT_RESET)) {
216 change_all_transmission_modes(blocks::UBlock::Transmission_Mode::ANALOG_INPUT);
218 reset_reference_magnitude();
222ConfigResult blocks::UBlock::config(
const pb_Config &cfg) {
223#ifdef ANABRID_DEBUG_ENTITY_CONFIG
224 Serial.println(__PRETTY_FUNCTION__);
226 if (cfg.which_kind != pb_Config_select_config_tag)
227 return ConfigResult::err(
"expected select block config");
229 auto& select_config = cfg.kind.select_config;
231 auto res = apply_outputs(select_config);
235 res = apply_constants(select_config);
239 return ConfigResult::ok(
true);
242ConfigResult blocks::UBlock::apply_outputs(
const pb_SelectConfig &cfg) {
246 for (
size_t idx = 0; idx < cfg.connections_count; ++idx) {
247 auto& connection = cfg.connections[idx];
248 if (!connect(connection.input, connection.output))
249 return ConfigResult::err(
"UBlock: Cannot connect input at idx");
251 return ConfigResult::ok(
true);
254ConfigResult blocks::UBlock::apply_constants(
const pb_SelectConfig &cfg) {
255 if (cfg.constant == pb_SelectConfig_ConstantConfig_GROUND) {
258 change_b_side_transmission_mode(blocks::UBlock::Transmission_Mode::ANALOG_INPUT);
259 reset_reference_magnitude();
260 return ConfigResult::ok(
true);
261 }
else if (cfg.constant == pb_SelectConfig_ConstantConfig_POS_REF) {
262 change_b_side_transmission_mode(blocks::UBlock::Transmission_Mode::POS_REF);
264 else if (cfg.constant == pb_SelectConfig_ConstantConfig_NEG_REF) {
265 change_b_side_transmission_mode(blocks::UBlock::Transmission_Mode::NEG_REF);
268 return ConfigResult::err_fmt(
"UBlock::config_self_from_pb: Ground, pos_ref or neg_ref");
270 if (cfg.magnitude == pb_SelectConfig_Magnitude_ONE_TENTH)
271 change_reference_magnitude(blocks::UBlock::Reference_Magnitude::ONE_TENTH);
272 else if (cfg.magnitude == pb_SelectConfig_Magnitude_ONE) {
273 change_reference_magnitude(blocks::UBlock::Reference_Magnitude::ONE);
276 return ConfigResult::err_fmt(
"UBlock::config_self_from_pb: Pass +-0.1 or +-1.");
279 return ConfigResult::ok(
true);
282void blocks::UBlock::extract(entities::ExtractVisitor &collector) {
283 Entity::extract(collector);
284 auto& cfg = collector.create(pb_Config_select_config_tag);
287 auto& select_config = cfg.kind.select_config = pb_SelectConfig_init_default;
288 for (
uint32_t idx = 0; idx < NUM_OF_OUTPUTS; ++idx) {
289 auto input = output_input_map[idx];
293 select_config.connections[select_config.connections_count++] = {
294 .input=
static_cast<uint32_t>(output_input_map[idx]),
302 switch (b_side_mode) {
303 case Transmission_Mode::POS_REF:
304 select_config.constant = pb_SelectConfig_ConstantConfig_POS_REF;
306 case Transmission_Mode::NEG_REF:
307 select_config.constant = pb_SelectConfig_ConstantConfig_NEG_REF;
310 select_config.constant = pb_SelectConfig_ConstantConfig_GROUND;
314 if (ref_magnitude == Reference_Magnitude::ONE_TENTH) {
315 select_config.magnitude = pb_SelectConfig_Magnitude_ONE_TENTH;
317 select_config.magnitude = pb_SelectConfig_Magnitude_ONE;
321void blocks::UBlock::reset_reference_magnitude() { ref_magnitude = Reference_Magnitude::ONE; }
323bool blocks::UBlockHAL_Dummy::write_outputs(std::array<int8_t, 32> outputs) {
return true; }
325bool blocks::UBlockHAL_Dummy::write_transmission_modes_and_ref(
326 std::pair<Transmission_Mode, Transmission_Mode> modes, blocks::UBlockHAL::Reference_Magnitude ref) {
330void blocks::UBlockHAL_Dummy::reset_transmission_modes_and_ref() {}