123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324 |
- #pragma once
- #include <stdint.h>
- #include <avr/pgmspace.h>
- template<typename T, uint8_t N>
- class array {
- T data[N];
- public:
- array() = default;
- inline constexpr T* begin()const { return data; }
- inline constexpr T* end()const { return data + N; }
- constexpr uint8_t size()const { return N; }
- inline T &operator[](uint8_t i){
- return data[i];
- }
- };
- #include "mmu2/error_codes.h"
- #include "mmu2/progress_codes.h"
- #include "mmu2/buttons.h"
- #include "mmu2_protocol.h"
- #include "mmu2_serial.h"
- namespace MMU2 {
- using namespace modules::protocol;
- class ProtocolLogic;
- enum StepStatus : uint_fast8_t {
- Processing = 0,
- MessageReady,
- Finished,
- CommunicationTimeout,
- ProtocolError,
- CommandRejected,
- CommandError,
- VersionMismatch,
- CommunicationRecovered,
- ButtonPushed,
- };
- static constexpr uint32_t linkLayerTimeout = 2000;
- static constexpr uint32_t dataLayerTimeout = linkLayerTimeout * 3;
- static constexpr uint32_t heartBeatPeriod = linkLayerTimeout / 2;
- static_assert(heartBeatPeriod < linkLayerTimeout && linkLayerTimeout < dataLayerTimeout, "Incorrect ordering of timeouts");
- class DropOutFilter {
- StepStatus cause;
- uint8_t occurrences;
- public:
- static constexpr uint8_t maxOccurrences = 10;
- static_assert(maxOccurrences > 1, "we should really silently ignore at least 1 comm drop out if recovered immediately afterwards");
- DropOutFilter() = default;
-
- bool Record(StepStatus ss);
-
- inline StepStatus InitialCause() const { return cause; }
-
- inline void Reset() { occurrences = maxOccurrences; }
- };
- class ProtocolLogic {
- public:
- ProtocolLogic(MMU2Serial *uart);
-
- void Start();
-
- void Stop();
-
- void ToolChange(uint8_t slot);
- void Statistics();
- void UnloadFilament();
- void LoadFilament(uint8_t slot);
- void EjectFilament(uint8_t slot);
- void CutFilament(uint8_t slot);
- void ResetMMU();
- void Button(uint8_t index);
- void Home(uint8_t mode);
- void ReadRegister(uint8_t address);
- void WriteRegister(uint8_t address, uint16_t data);
-
- StepStatus Step();
-
- ErrorCode Error() const { return errorCode; }
-
- ProgressCode Progress() const { return progressCode; }
-
- Buttons Button() const { return buttonCode; }
- uint8_t CommandInProgress() const;
- inline bool Running() const {
- return state == State::Running;
- }
- inline bool FindaPressed() const {
- return regs8[0];
- }
- inline uint16_t FailStatistics() const {
- return regs16[0];
- }
- inline uint8_t MmuFwVersionMajor() const {
- return mmuFwVersion[0];
- }
- inline uint8_t MmuFwVersionMinor() const {
- return mmuFwVersion[1];
- }
- inline uint8_t MmuFwVersionRevision() const {
- return mmuFwVersion[2];
- }
- #ifndef UNITTEST
- private:
- #endif
- StepStatus ExpectingMessage();
- void SendMsg(RequestMsg rq);
- void SendWriteMsg(RequestMsg rq);
- void SwitchToIdle();
- StepStatus SuppressShortDropOuts(const char *msg_P, StepStatus ss);
- StepStatus HandleCommunicationTimeout();
- StepStatus HandleProtocolError();
- bool Elapsed(uint32_t timeout) const;
- void RecordUARTActivity();
- void RecordReceivedByte(uint8_t c);
- void FormatLastReceivedBytes(char *dst);
- void FormatLastResponseMsgAndClearLRB(char *dst);
- void LogRequestMsg(const uint8_t *txbuff, uint8_t size);
- void LogError(const char *reason_P);
- void LogResponse();
- StepStatus SwitchFromIdleToCommand();
- void SwitchFromStartToIdle();
- enum class State : uint_fast8_t {
- Stopped,
- InitSequence,
- Running
- };
-
-
-
- enum class Scope : uint_fast8_t {
- Stopped,
- StartSeq,
- DelayedRestart,
- Idle,
- Command
- };
- Scope currentScope;
-
-
- bool ExpectsResponse() const { return ((uint8_t)scopeState & (uint8_t)ScopeState::NotExpectsResponse) == 0; }
-
-
- enum class ScopeState : uint_fast8_t {
- S0Sent,
- S1Sent,
- S2Sent,
- S3Sent,
- QuerySent,
- CommandSent,
- FilamentSensorStateSent,
- Reading8bitRegisters,
- Reading16bitRegisters,
- ButtonSent,
- ReadRegisterSent,
- WriteRegisterSent,
-
- NotExpectsResponse = 0x80,
- Wait = NotExpectsResponse + 1,
- Ready = NotExpectsResponse + 2,
- RecoveringProtocolError = NotExpectsResponse + 3,
- };
- ScopeState scopeState;
-
-
-
-
-
-
-
-
-
-
-
- void CheckAndReportAsyncEvents();
- void SendQuery();
- void StartReading8bitRegisters();
- void ProcessRead8bitRegister();
- void StartReading16bitRegisters();
- ScopeState ProcessRead16bitRegister(ProtocolLogic::ScopeState stateAtEnd);
- void SendAndUpdateFilamentSensor();
- void SendButton(uint8_t btn);
- void SendVersion(uint8_t stage);
- void SendReadRegister(uint8_t index, ScopeState nextState);
- void SendWriteRegister(uint8_t index, uint16_t value, ScopeState nextState);
- StepStatus ProcessVersionResponse(uint8_t stage);
-
- StepStatus ScopeStep();
- static constexpr uint8_t maxRetries = 6;
- uint8_t retries;
- void StartSeqRestart();
- void DelayedRestartRestart();
- void IdleRestart();
- void CommandRestart();
- StepStatus StartSeqStep();
- StepStatus DelayedRestartWait();
- StepStatus IdleStep();
- StepStatus IdleWait();
- StepStatus CommandStep();
- StepStatus CommandWait();
- StepStatus StoppedStep() { return Processing; }
- StepStatus ProcessCommandQueryResponse();
- inline void SetRequestMsg(RequestMsg msg) {
- rq = msg;
- }
- inline const RequestMsg &ReqMsg() const { return rq; }
- RequestMsg rq = RequestMsg(RequestMsgCodes::unknown, 0);
-
-
-
-
-
-
-
-
-
-
- RequestMsg plannedRq;
-
- void PlanGenericRequest(RequestMsg rq);
-
- bool ActivatePlannedRequest();
- uint32_t lastUARTActivityMs;
- DropOutFilter dataTO;
- ResponseMsg rsp;
- State state;
- Protocol protocol;
- array<uint8_t, 16> lastReceivedBytes;
- uint8_t lrb;
- MMU2Serial *uart;
- ErrorCode errorCode;
- ProgressCode progressCode;
- Buttons buttonCode;
- uint8_t lastFSensor;
-
- static constexpr uint8_t regs8Count = 3;
- static_assert(regs8Count > 0);
- static const uint8_t regs8Addrs[regs8Count] PROGMEM;
- uint8_t regs8[regs8Count];
-
- static constexpr uint8_t regs16Count = 2;
- static_assert(regs16Count > 0);
- static const uint8_t regs16Addrs[regs16Count] PROGMEM;
- uint16_t regs16[regs16Count];
- uint8_t regIndex;
- uint8_t mmuFwVersion[3];
- uint16_t mmuFwVersionBuild;
- friend class ProtocolLogicPartBase;
- friend class Stopped;
- friend class Command;
- friend class Idle;
- friend class StartSeq;
- friend class DelayedRestart;
- friend class MMU2;
- };
- }
|