diff --git a/VortexEngine/src/Menus/MenuList/EditorConnection.cpp b/VortexEngine/src/Menus/MenuList/EditorConnection.cpp index 5bf5786cea..ebf6851651 100644 --- a/VortexEngine/src/Menus/MenuList/EditorConnection.cpp +++ b/VortexEngine/src/Menus/MenuList/EditorConnection.cpp @@ -188,6 +188,89 @@ Menu::MenuAction EditorConnection::run() SerialComs::write(EDITOR_VERB_TRANSMIT_VL_ACK); m_state = STATE_IDLE; break; + case STATE_PULL_EACH_MODE: + // editor requested pull modes, send the modes + m_receiveBuffer.clear(); + sendModeCount(); + m_state = STATE_PULL_EACH_MODE_COUNT; + break; + case STATE_PULL_EACH_MODE_COUNT: + if (receiveMessage(EDITOR_VERB_PULL_EACH_MODE_ACK)) { + if (Modes::numModes() == 0) { + m_state = STATE_PULL_EACH_MODE_DONE; + } else { + m_previousModeIndex = Modes::curModeIndex(); + m_state = STATE_PULL_EACH_MODE_SEND; + } + } + break; + case STATE_PULL_EACH_MODE_SEND: + m_receiveBuffer.clear(); + // send the current mode + sendCurMode(); + // wait for the ack + m_state = STATE_PULL_EACH_MODE_WAIT; + break; + case STATE_PULL_EACH_MODE_WAIT: + // recive the ack from the editor to send next mode + if (receiveMessage(EDITOR_VERB_PULL_EACH_MODE_ACK)) { + // if there is still more modes + if (Modes::curModeIndex() < (Modes::numModes() - 1)) { + // then iterate to the next mode and send + Modes::nextMode(); + m_state = STATE_PULL_EACH_MODE_SEND; + } else { + // otherwise done sending modes + m_state = STATE_PULL_EACH_MODE_DONE; + } + } + break; + case STATE_PULL_EACH_MODE_DONE: + m_receiveBuffer.clear(); + // send our acknowledgement that the modes were sent + SerialComs::write(EDITOR_VERB_PULL_EACH_MODE_DONE); + // switch back to the previous mode + Modes::setCurMode(m_previousModeIndex); + // go idle + m_state = STATE_IDLE; + break; + case STATE_PUSH_EACH_MODE: + // editor requested to push modes, find out how many + m_receiveBuffer.clear(); + // ack the command and wait for the amount of modes + SerialComs::write(EDITOR_VERB_PUSH_EACH_MODE_ACK); + m_state = STATE_PUSH_EACH_MODE_COUNT; + break; + case STATE_PUSH_EACH_MODE_COUNT: + if (receiveModeCount()) { + // clear modes and start receiving + Modes::clearModes(); + // write out an ack + m_receiveBuffer.clear(); + SerialComs::write(EDITOR_VERB_PUSH_EACH_MODE_ACK); + // ready to receive a mode + m_state = STATE_PUSH_EACH_MODE_RECEIVE; + } + break; + case STATE_PUSH_EACH_MODE_RECEIVE: + // receive the modes into the receive buffer + if (receiveMode()) { + if (Modes::numModes() < m_numModesToReceive) { + // clear the receive buffer and ack the mode, continue receiving + m_receiveBuffer.clear(); + SerialComs::write(EDITOR_VERB_PUSH_EACH_MODE_ACK); + } else { + // success modes were received send the done + m_state = STATE_PUSH_EACH_MODE_DONE; + } + } + break; + case STATE_PUSH_EACH_MODE_DONE: + // say we are done + m_receiveBuffer.clear(); + SerialComs::write(EDITOR_VERB_PUSH_EACH_MODE_DONE); + m_state = STATE_IDLE; + break; } return MENU_CONTINUE; } @@ -264,6 +347,28 @@ void EditorConnection::sendModes() SerialComs::write(modesBuffer); } +void EditorConnection::sendModeCount() +{ + ByteStream buffer; + buffer.serialize8(Modes::numModes()); + SerialComs::write(buffer); +} + +void EditorConnection::sendCurMode() +{ + ByteStream modeBuffer; + Mode *cur = Modes::curMode(); + if (!cur) { + // ?? + return; + } + if (!cur->saveToBuffer(modeBuffer)) { + // ?? + return; + } + SerialComs::write(modeBuffer); +} + bool EditorConnection::receiveModes() { // need at least the buffer size first @@ -296,6 +401,76 @@ bool EditorConnection::receiveModes() return true; } +bool EditorConnection::receiveModeCount() +{ + // need at least the buffer size first + uint32_t size = 0; + if (m_receiveBuffer.size() < sizeof(size)) { + // wait, not enough data available yet + return false; + } + // grab the size out of the start + m_receiveBuffer.resetUnserializer(); + size = m_receiveBuffer.peek32(); + if (m_receiveBuffer.size() < (size + sizeof(size))) { + // don't unserialize yet, not ready + return false; + } + // okay unserialize now, first unserialize the size + if (!m_receiveBuffer.unserialize32(&size)) { + return false; + } + // create a new ByteStream that will hold the full buffer of data + ByteStream buf(m_receiveBuffer.rawSize()); + // then copy everything from the receive buffer into the rawdata + // which is going to overwrite the crc/size/flags of the ByteStream + memcpy(buf.rawData(), m_receiveBuffer.data() + sizeof(size), + m_receiveBuffer.size() - sizeof(size)); + // unserialize the mode count + if (!buf.unserialize8(&m_numModesToReceive)) { + return false; + } + if (m_numModesToReceive > MAX_MODES) { + return false; + } + // good mode count + return true; +} + +bool EditorConnection::receiveMode() +{ + // need at least the buffer size first + uint32_t size = 0; + if (m_receiveBuffer.size() < sizeof(size)) { + // wait, not enough data available yet + return false; + } + // grab the size out of the start + m_receiveBuffer.resetUnserializer(); + size = m_receiveBuffer.peek32(); + if (m_receiveBuffer.size() < (size + sizeof(size))) { + // don't unserialize yet, not ready + return false; + } + // okay unserialize now, first unserialize the size + if (!m_receiveBuffer.unserialize32(&size)) { + return false; + } + // create a new ByteStream that will hold the full buffer of data + ByteStream buf(m_receiveBuffer.rawSize()); + // then copy everything from the receive buffer into the rawdata + // which is going to overwrite the crc/size/flags of the ByteStream + memcpy(buf.rawData(), m_receiveBuffer.data() + sizeof(size), + m_receiveBuffer.size() - sizeof(size)); + // clear the receive buffer + m_receiveBuffer.clear(); + // unserialize the mode into the demo mode + if (!Modes::addModeFromBuffer(buf)) { + // error + } + return true; +} + bool EditorConnection::receiveDemoMode() { // need at least the buffer size first @@ -340,6 +515,10 @@ void EditorConnection::handleCommand() m_state = STATE_DEMO_MODE; } else if (receiveMessage(EDITOR_VERB_CLEAR_DEMO)) { m_state = STATE_CLEAR_DEMO; + } else if (receiveMessage(EDITOR_VERB_PULL_EACH_MODE)) { + m_state = STATE_PULL_EACH_MODE; + } else if (receiveMessage(EDITOR_VERB_PUSH_EACH_MODE)) { + m_state = STATE_PUSH_EACH_MODE; } else if (receiveMessage(EDITOR_VERB_TRANSMIT_VL)) { sendCurModeVL(); } diff --git a/VortexEngine/src/Menus/MenuList/EditorConnection.h b/VortexEngine/src/Menus/MenuList/EditorConnection.h index 31edfa444f..56f5927100 100644 --- a/VortexEngine/src/Menus/MenuList/EditorConnection.h +++ b/VortexEngine/src/Menus/MenuList/EditorConnection.h @@ -29,7 +29,11 @@ class EditorConnection : public Menu void showEditor(); void receiveData(); void sendModes(); + void sendModeCount(); + void sendCurMode(); bool receiveModes(); + bool receiveModeCount(); + bool receiveMode(); bool receiveDemoMode(); void handleCommand(); bool receiveMessage(const char *message); @@ -39,7 +43,7 @@ class EditorConnection : public Menu virtual void showExit() override; enum EditorConnectionState { - // the editor is not connec + // the editor is not connected STATE_DISCONNECTED, // Sending the greeting message @@ -48,27 +52,41 @@ class EditorConnection : public Menu // entirely idle, waiting for commands STATE_IDLE, - // engine pulls the modes from gloves, then wait for the sent modes ack + // editor pulls the modes from device, then wait for the sent modes ack STATE_PULL_MODES, STATE_PULL_MODES_SEND, STATE_PULL_MODES_DONE, - // engine pushes modes to gloves, then waits for done + // editor pushes modes to device, then waits for done STATE_PUSH_MODES, STATE_PUSH_MODES_RECEIVE, STATE_PUSH_MODES_DONE, - // engine pushes mode to gloves for demo while idle + // editor pushes mode to device for demo while idle STATE_DEMO_MODE, STATE_DEMO_MODE_RECEIVE, STATE_DEMO_MODE_DONE, - // engine tells gloves to clear the demo preview, gloves acknowledge + // editor tells device to clear the demo preview, device acknowledge STATE_CLEAR_DEMO, // transmit the mode over visible light STATE_TRANSMIT_MODE_VL, STATE_TRANSMIT_MODE_VL_DONE, + + // editor pulls the modes from device (safer version) + STATE_PULL_EACH_MODE, + STATE_PULL_EACH_MODE_COUNT, + STATE_PULL_EACH_MODE_SEND, + STATE_PULL_EACH_MODE_WAIT, + STATE_PULL_EACH_MODE_DONE, + + // editor pushes modes to device (safer version) + STATE_PUSH_EACH_MODE, + STATE_PUSH_EACH_MODE_COUNT, + STATE_PUSH_EACH_MODE_RECEIVE, + STATE_PUSH_EACH_MODE_WAIT, + STATE_PUSH_EACH_MODE_DONE, }; // state of the editor @@ -77,6 +95,10 @@ class EditorConnection : public Menu ByteStream m_receiveBuffer; // Whether at least one command has been received yet bool m_allowReset; + // the mode index to return to after iterating the modes to send them + uint8_t m_previousModeIndex; + // the number of modes that should be received + uint8_t m_numModesToReceive; }; #endif