Files
-
blinds / hardware_design / pcb / blinds_v60.brd
-
blinds / hardware_design / pcb / blinds_v60.sch
-
braids / hardware_design / pcb / braids_v50.brd
-
braids / hardware_design / pcb / braids_v50.sch
-
branches / hardware_design / pcb / branches_v40.brd
-
branches / hardware_design / pcb / branches_v40.sch
-
clouds / hardware_design / pcb / clouds_v30.brd
-
clouds / hardware_design / pcb / clouds_v30.sch
-
ears / hardware_design / panel / ears_panel_v30.brd
-
ears / hardware_design / panel / ears_panel_v30.sch
-
ears / hardware_design / pcb / ears_v40.brd
-
ears / hardware_design / pcb / ears_v40.sch
-
edges / hardware_design / pcb / edges_expander_v01.brd
-
edges / hardware_design / pcb / edges_expander_v01.sch
-
edges / hardware_design / pcb / edges_v20.brd
-
edges / hardware_design / pcb / edges_v20.sch
-
elements / hardware_design / pcb / elements_v02.brd
-
elements / hardware_design / pcb / elements_v02.sch
-
frames / hardware_design / pcb / frames_v03.brd
-
frames / hardware_design / pcb / frames_v03.sch
-
grids / hardware_design / pcb / grids_v02.brd
-
grids / hardware_design / pcb / grids_v02.sch
-
kinks / hardware_design / pcb / kinks_v41.brd
-
kinks / hardware_design / pcb / kinks_v41.sch
-
links / hardware_design / pcb / links_v40.brd
-
links / hardware_design / pcb / links_v40.sch
-
marbles / hardware_design / pcb / marbles_v70.brd
-
marbles / hardware_design / pcb / marbles_v70.sch
-
peaks / hardware_design / pcb / peaks_v30.brd
-
peaks / hardware_design / pcb / peaks_v30.sch
-
plaits / hardware_design / pcb / plaits_v50.brd
-
plaits / hardware_design / pcb / plaits_v50.sch
-
rings / hardware_design / pcb / rings_v30.brd
-
rings / hardware_design / pcb / rings_v30.sch
-
ripples / hardware_design / pcb / ripples_v40.brd
-
ripples / hardware_design / pcb / ripples_v40.sch
-
shades / hardware_design / pcb / shades_v30.brd
-
shades / hardware_design / pcb / shades_v30.sch
-
shelves / hardware_design / pcb / shelves_expander_v10.brd
-
shelves / hardware_design / pcb / shelves_expander_v10.sch
-
shelves / hardware_design / pcb / shelves_v05.brd
-
shelves / hardware_design / pcb / shelves_v05.sch
-
stages / hardware_design / pcb / stages_v70.brd
-
stages / hardware_design / pcb / stages_v70.sch
-
streams / hardware_design / pcb / streams_v02_bargraph.brd
-
streams / hardware_design / pcb / streams_v02_bargraph.sch
-
streams / hardware_design / pcb / streams_v05.brd
-
streams / hardware_design / pcb / streams_v05.sch
-
tides / hardware_design / pcb / tides_v40.brd
-
tides / hardware_design / pcb / tides_v40.sch
-
veils / hardware_design / pcb / veils_v40.brd
-
veils / hardware_design / pcb / veils_v40.sch
-
volts / hardware_design / pcb / volts_v01.brd
-
volts / hardware_design / pcb / volts_v01.sch
-
warps / hardware_design / pcb / warps_v30.brd
-
warps / hardware_design / pcb / warps_v30.sch
-
yarns / hardware_design / pcb / yarns_v03.brd
-
yarns / hardware_design / pcb / yarns_v03.sch
Last update 6 years 3 months
by
Olivier Gillet
chain_state.cc// Copyright 2017 Olivier Gillet. // // Author: Olivier Gillet (ol.gillet@gmail.com) // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to deal // in the Software without restriction, including without limitation the rights // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell // copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN // THE SOFTWARE. // // See http://creativecommons.org/licenses/MIT/ for more information. // // ----------------------------------------------------------------------------- // // Chain state. #include "stages/chain_state.h" #include <algorithm> #include "stages/drivers/serial_link.h" #include "stages/settings.h" namespace stages { using namespace std; const uint32_t kLeftKey = stmlib::FourCC<'d', 'i', 's', 'c'>::value; const uint32_t kRightKey = stmlib::FourCC<'o', 'v', 'e', 'r'>::value; // How long before unpatching an input actually breaks the chain. const uint32_t kUnpatchedInputDelay = 2000; const int32_t kLongPressDuration = 800; void ChainState::Init(SerialLink* left, SerialLink* right) { index_ = 0; size_ = 1; left_ = left; right_ = right; STATIC_ASSERT(sizeof(Packet) == kPacketSize, BAD_PACKET_SIZE); left_->Init( SERIAL_LINK_DIRECTION_LEFT, 115200 * 8, left_rx_packet_[0].bytes, kPacketSize); right_->Init( SERIAL_LINK_DIRECTION_RIGHT, 115200 * 8, right_rx_packet_[0].bytes, kPacketSize); ChannelState c = { .flags = 0xf0, .pot = 128, .cv_slider = 32768 }; fill(&channel_state_[0], &channel_state_[kMaxNumChannels], c); fill(&unpatch_counter_[0], &unpatch_counter_[kNumChannels], 0); fill(&loop_status_[0], &loop_status_[kNumChannels], LOOP_STATUS_NONE); fill(&switch_pressed_[0], &switch_pressed_[kMaxChainSize], 0); fill(&switch_press_time_[0], &switch_press_time_[kMaxNumChannels], 0); request_.request = REQUEST_NONE; discovering_neighbors_ = true; ouroboros_ = false; counter_ = 0; num_internal_bindings_ = 0; num_bindings_ = 0; } void ChainState::DiscoverNeighbors() { // Between t = 500ms and t = 1500ms, ping the neighbors every 50ms if (counter_ >= 2000 && counter_ <= 6000 && (counter_ % 200) == 0) { left_tx_packet_.discovery.key = kLeftKey; left_tx_packet_.discovery.counter = size_; left_->Transmit(left_tx_packet_); right_tx_packet_.discovery.key = kRightKey; right_tx_packet_.discovery.counter = index_; right_->Transmit(right_tx_packet_); } const DiscoveryPacket* l = left_->available_rx_buffer<DiscoveryPacket>(); if (l && l->key == kRightKey) { index_ = size_t(l->counter) + 1; size_ = std::max(size_, index_ + 1); } const DiscoveryPacket* r = right_->available_rx_buffer<DiscoveryPacket>(); if (r && r->key == kLeftKey) { size_ = std::max(size_, size_t(r->counter)); } ouroboros_ = index_ >= kMaxChainSize || size_ > kMaxChainSize; // The discovery phase lasts 2000ms. discovering_neighbors_ = counter_ < 8000 && !ouroboros_; if (discovering_neighbors_) { ++counter_; } else { counter_ = 0; } } void ChainState::TransmitRight() { if (index_ == size_ - 1) { return; } LeftToRightPacket* p = &right_tx_packet_.to_right; p->phase = tx_last_sample_.phase; p->segment = tx_last_sample_.segment; p->last_patched_channel = tx_last_patched_channel_; p->last_loop = tx_last_loop_; copy(&input_patched_[0], &input_patched_[index_ + 1], &p->input_patched[0]); copy(&switch_pressed_[0], &switch_pressed_[index_ + 1], &p->switch_pressed[0]); right_->Transmit(right_tx_packet_); } void ChainState::ReceiveRight() { if (index_ == size_ - 1) { return; } const RightToLeftPacket* p = right_->available_rx_buffer<RightToLeftPacket>(); if (p) { size_t rx_index = p->channel[0].index(); if (rx_index > index_ && rx_index < size_) { // This packet contains the state of a module on the right. // Check if some settings have been changed on the remote modules, // then update our local copy of its state. for (size_t i = 0; i < kNumChannels; ++i) { dirty_[remote_channel_index(rx_index, i)] = \ remote_channel(rx_index, i)->flags != p->channel[i].flags; } copy( &p->channel[0], &p->channel[kNumChannels], remote_channel(rx_index, 0)); request_.request = REQUEST_NONE; } else if (rx_index == 0xf) { // This suspiciously looks like a state change request packet! // We will take care of it later. request_ = *(const RequestPacket*)(p); } } } void ChainState::TransmitLeft() { if (index_ == 0) { return; } if (request_.request != REQUEST_NONE) { // Forward the request to the left. left_tx_packet_.request = request_; } else { // Determine which module contains the last segment of the chain // starting at this module's last segment. // // For example: // // 0----- 1----- 2----- 3----- 4----- // // ----X- ------ ---X-- ------ ---X-- // // last = 2 // last = 2 // last = 4 // last = 4 // last = 5 size_t last = size_ - 1; for (size_t i = index_; i < size_; ++i) { for (size_t j = 0; j < kNumChannels; ++j) { if (remote_channel(i, j)->input_patched()) { last = i; goto found; } } } found: // In the example above, module 1 will alternate between sending to module // 0 its own state, and the state of module 2. size_t tx_index = index_ + ((counter_ >> 2) % (last - index_ + 1)); copy( remote_channel(tx_index, 0), remote_channel(tx_index, kNumChannels), &left_tx_packet_.to_left.channel[0]); } left_->Transmit(left_tx_packet_); } void ChainState::ReceiveLeft() { if (index_ == 0) { rx_last_patched_channel_ = size_ * kNumChannels; rx_last_loop_.start = -1; rx_last_loop_.end = -1; return; } const LeftToRightPacket* p = left_->available_rx_buffer<LeftToRightPacket>(); if (p) { rx_last_patched_channel_ = p->last_patched_channel; rx_last_loop_ = p->last_loop; rx_last_sample_.phase = p->phase; rx_last_sample_.segment = p->segment; copy(&p->switch_pressed[0], &p->switch_pressed[index_], &switch_pressed_[0]); copy(&p->input_patched[0], &p->input_patched[index_], &input_patched_[0]); } } void ChainState::Configure(SegmentGenerator* segment_generator) { size_t last_local_channel = local_channel_index(0) + kNumChannels; size_t last_channel = size_ * kNumChannels; size_t last_patched_channel = rx_last_patched_channel_; Loop last_loop = rx_last_loop_; num_internal_bindings_ = 0; num_bindings_ = 0; segment::Configuration configuration[kMaxNumChannels]; for (size_t i = 0; i < kNumChannels; ++i) { size_t channel = local_channel_index(i); if (!local_channel(i)->input_patched()) { if (channel > last_patched_channel) { // Create a slave channel - we are just extending a chain of segments. size_t segment = channel - last_patched_channel; segment_generator[i].ConfigureSlave(segment); set_loop_status(i, segment, last_loop); } else { // Create a free-running channel. segment::Configuration c = local_channel(i)->configuration(); segment_generator[i].ConfigureSingleSegment(false, c); binding_[num_bindings_].generator = i; binding_[num_bindings_].source = i; binding_[num_bindings_].destination = 0; ++num_bindings_; ++num_internal_bindings_; loop_status_[i] = c.loop ? LOOP_STATUS_SELF : LOOP_STATUS_NONE; } } else { last_patched_channel = channel; // Create a normal channel, trying to extend it as far as possible. int num_segments = 0; bool add_more_segments = true; bool dirty = false; last_loop.start = -1; last_loop.end = -1; while (add_more_segments) { // Add an entry in the configuration array. segment::Configuration c = channel_state_[channel].configuration(); configuration[num_segments] = c; dirty |= dirty_[channel]; if (c.loop) { if (last_loop.start == -1) last_loop.start = num_segments; last_loop.end = num_segments; } // Add a binding in the binding array. binding_[num_bindings_].generator = i; binding_[num_bindings_].destination = num_segments; if (channel < last_local_channel) { // Bind local CV/pot to this segment's parameters. binding_[num_bindings_].source = i + num_segments; ++num_internal_bindings_; } else { // Bind remote CV/pot to this segment's parameters. binding_[num_bindings_].source = channel; } ++num_bindings_; ++channel; ++num_segments; add_more_segments = channel < last_channel && \ !channel_state_[channel].input_patched(); } if (dirty || num_segments != segment_generator[i].num_segments()) { segment_generator[i].Configure(true, configuration, num_segments); } set_loop_status(i, 0, last_loop); } } tx_last_loop_ = last_loop; tx_last_patched_channel_ = last_patched_channel; } inline void ChainState::UpdateLocalState( const IOBuffer::Block& block, const Settings& settings, const SegmentGenerator::Output& last_out) { tx_last_sample_ = last_out; ChannelBitmask input_patched_bitmask = 0; for (size_t i = 0; i < kNumChannels; ++i) { if (block.input_patched[i]) { unpatch_counter_[i] = 0; } else if (unpatch_counter_[i] < kUnpatchedInputDelay) { ++unpatch_counter_[i]; } bool input_patched = unpatch_counter_[i] < kUnpatchedInputDelay; dirty_[local_channel_index(i)] = local_channel(i)->UpdateFlags( index_, settings.state().segment_configuration[i], input_patched); if (input_patched) { input_patched_bitmask |= 1 << i; } } input_patched_[index_] = input_patched_bitmask; } inline void ChainState::UpdateLocalPotCvSlider(const IOBuffer::Block& block) { for (size_t i = 0; i < kNumChannels; ++i) { ChannelState* s = local_channel(i); s->pot = block.pot[i] * 256.0f; s->cv_slider = block.cv_slider[i] * 16384.0f + 32768.0f; } } inline void ChainState::BindRemoteParameters( SegmentGenerator* segment_generator) { for (size_t i = num_internal_bindings_; i < num_bindings_; ++i) { const ParameterBinding& m = binding_[i]; segment_generator[m.generator].set_segment_parameters( m.destination, channel_state_[m.source].cv_slider / 16384.0f - 2.0f, channel_state_[m.source].pot / 256.0f); } } inline void ChainState::BindLocalParameters( const IOBuffer::Block& block, SegmentGenerator* segment_generator) { for (size_t i = 0; i < num_internal_bindings_; ++i) { const ParameterBinding& m = binding_[i]; segment_generator[m.generator].set_segment_parameters( m.destination, block.cv_slider[m.source], block.pot[m.source]); } } ChainState::RequestPacket ChainState::MakeLoopChangeRequest( size_t loop_start, size_t loop_end) { size_t channel_index = 0; size_t group_start = 0; size_t group_end = size_ * kNumChannels; bool inconsistent_loop = false; // Fill group_start and group_end, which contain the tightest interval // of patched channels enclosing the loop. // // LOOP ----S- ------ --E--- ------ // PATCHED -x---- ------ ----x- ------ // ^ ^ ^ ^ // | | | | // | | | group_end // | | loop_end // | loop_start // group_start for (size_t i = 0; i < size_; ++i) { ChannelBitmask input_patched = input_patched_[i]; for (size_t j = 0; j < kNumChannels; ++j) { if (input_patched & 1) { if (channel_index <= loop_start) { group_start = channel_index; } else if (channel_index >= loop_end) { group_end = min(group_end, channel_index); } // There shouldn't be a patched channel between the loop start // and the loop end. if (channel_index > loop_start && channel_index < loop_end) { // LOOP ----S- ------ --E--- ------ // PATCHED -x---- ---x-- ----x- ------ inconsistent_loop = true; } } input_patched >>= 1; ++channel_index; } } // There shouldn't be a loop spanning multiple channels among the first // group of unpatched channels. if (group_start == 0 && !(input_patched_[0] & 1)) { if (loop_start != loop_end) { // LOOP -S-E-- ------ // PATCHED -----x ---x-- inconsistent_loop = true; } else { group_start = group_end = loop_start = loop_end; } } // The only situation where a loop can end on a patched channel is when // we have a single-channel group. if (group_end == loop_end && group_start != group_end) { // Correct: // LOOP ---S-- // E // PATCHED ---xx- // Incorrect: // LOOP ---S-E // PATCHED ---x-x inconsistent_loop = true; } RequestPacket result; if (inconsistent_loop) { result.request = REQUEST_NONE; } else { result.request = REQUEST_SET_LOOP; result.argument[0] = group_start; result.argument[1] = loop_start; result.argument[2] = loop_end; result.argument[3] = group_end; } return result; } void ChainState::PollSwitches() { // The last module in the chain polls the states of the switches for the // entire chain. The state of the switches has been passed from left // to right. // // If a switch has been pressed, a Request packet is passed from right // to left. Each module is responsible from parsing the Request packet // and adjusting its internal state to simulate local changes. During // the next cycle (1ms later), the internal change will be propagated // from module to module through the usual mechanism (ChannelState // transmission). if (index_ == size_ - 1) { request_.request = REQUEST_NONE; size_t switch_index = 0; size_t first_pressed = kMaxNumChannels; for (size_t i = 0; i < size_; ++i) { ChannelBitmask switch_pressed = switch_pressed_[i]; for (size_t j = 0; j < kNumChannels; ++j) { if (switch_pressed & 1) { if (switch_press_time_[switch_index] != -1) { ++switch_press_time_[switch_index]; if (first_pressed != kMaxNumChannels) { // Simultaneously pressing a pair of buttons. request_ = MakeLoopChangeRequest(first_pressed, switch_index); switch_press_time_[first_pressed] = -1; switch_press_time_[switch_index] = -1; } else if (switch_press_time_[switch_index] > kLongPressDuration) { // Long press on a single button. request_ = MakeLoopChangeRequest(switch_index, switch_index); switch_press_time_[switch_index] = -1; } else { first_pressed = switch_index; } } } else { if (switch_press_time_[switch_index] > 5) { // A button has been released after having been held for a // sufficiently long time (5ms), but not for long enough to be // detected as a long press. request_.request = REQUEST_SET_SEGMENT_TYPE; request_.argument[0] = switch_index; } switch_press_time_[switch_index] = 0; } switch_pressed >>= 1; ++switch_index; } } } } void ChainState::HandleRequest(Settings* settings) { if (request_.request == REQUEST_NONE) { return; } State* s = settings->mutable_state(); bool dirty = false; for (size_t i = 0; i < kNumChannels; ++i) { size_t channel = local_channel_index(i); uint8_t type_bits = s->segment_configuration[i] & 0x3; uint8_t loop_bit = s->segment_configuration[i] & 0x4; if (request_.request == REQUEST_SET_SEGMENT_TYPE) { if (channel == request_.argument[0]) { s->segment_configuration[i] = ((type_bits + 1) % 3) | loop_bit; dirty |= true; } } else if (request_.request == REQUEST_SET_LOOP) { uint8_t new_loop_bit = loop_bit; if ((channel >= request_.argument[0] && channel < request_.argument[3])) { new_loop_bit = 0x0; } if (channel == request_.argument[1] || channel == request_.argument[2]) { if (request_.argument[1] == request_.argument[2]) { new_loop_bit = 0x4 - loop_bit; } else { new_loop_bit = 0x4; } } s->segment_configuration[i] = type_bits | new_loop_bit; dirty |= new_loop_bit != loop_bit; } } if (dirty) { settings->SaveState(); } } void ChainState::Update( const IOBuffer::Block& block, Settings* settings, SegmentGenerator* segment_generator, SegmentGenerator::Output* out) { if (discovering_neighbors_) { DiscoverNeighbors(); return; } switch (counter_ & 0x3) { case 0: PollSwitches(); UpdateLocalState(block, *settings, out[kBlockSize - 1]); TransmitRight(); break; case 1: ReceiveRight(); HandleRequest(settings); break; case 2: UpdateLocalPotCvSlider(block); TransmitLeft(); break; case 3: ReceiveLeft(); Configure(segment_generator); BindRemoteParameters(segment_generator); break; } BindLocalParameters(block, segment_generator); fill(&out[0], &out[kBlockSize], rx_last_sample_); ++counter_; } } // namespace stages