Fix and improve G-code queue (#21122)
Co-authored-by: Scott Lahteine <thinkyhead@users.noreply.github.com>
This commit is contained in:
		| @@ -407,7 +407,7 @@ void startOrResumeJob() { | ||||
|  */ | ||||
| inline void manage_inactivity(const bool ignore_stepper_queue=false) { | ||||
|  | ||||
|   if (queue.length < BUFSIZE) queue.get_available_commands(); | ||||
|   queue.get_available_commands(); | ||||
|  | ||||
|   const millis_t ms = millis(); | ||||
|  | ||||
|   | ||||
							
								
								
									
										37
									
								
								Marlin/src/core/bug_on.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								Marlin/src/core/bug_on.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,37 @@ | ||||
| /** | ||||
|  * Marlin 3D Printer Firmware | ||||
|  * Copyright (c) 2021 MarlinFirmware [https://github.com/MarlinFirmware/Marlin] | ||||
|  * | ||||
|  * Copyright (c) 2021 X-Ryl669 [https://blog.cyril.by] | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU General Public License as published by | ||||
|  * the Free Software Foundation, either version 3 of the License, or | ||||
|  * (at your option) any later version. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU General Public License for more details. | ||||
|  * | ||||
|  * You should have received a copy of the GNU General Public License | ||||
|  * along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  * | ||||
|  */ | ||||
| #pragma once | ||||
|  | ||||
| // We need SERIAL_ECHOPAIR and macros.h | ||||
| #include "serial.h" | ||||
|  | ||||
| #if ENABLED(POSTMORTEM_DEBUGGING) | ||||
|   // Useful macro for stopping the CPU on an unexpected condition | ||||
|   // This is used like SERIAL_ECHOPAIR, that is: a key-value call of the local variables you want | ||||
|   // to dump to the serial port before stopping the CPU. | ||||
|   #define BUG_ON(V...) do { SERIAL_ECHOPAIR(ONLY_FILENAME, __LINE__, ": "); SERIAL_ECHOLNPAIR(V); SERIAL_FLUSHTX(); *(char*)0 = 42; } while(0) | ||||
| #elif ENABLED(MARLIN_DEV_MODE) | ||||
|   // Don't stop the CPU here, but at least dump the bug on the serial port | ||||
|   #define BUG_ON(V...) do { SERIAL_ECHOPAIR(ONLY_FILENAME, __LINE__, ": BUG!\n"); SERIAL_ECHOLNPAIR(V); SERIAL_FLUSHTX(); } while(0) | ||||
| #else | ||||
|   // Release mode, let's ignore the bug | ||||
|   #define BUG_ON(V...) NOOP | ||||
| #endif | ||||
| @@ -131,6 +131,7 @@ | ||||
| #define STR_WATCHDOG_FIRED                  "Watchdog timeout. Reset required." | ||||
| #define STR_ERR_KILLED                      "Printer halted. kill() called!" | ||||
| #define STR_ERR_STOPPED                     "Printer stopped due to errors. Fix the error and use M999 to restart. (Temperature is reset. Set it after restarting)" | ||||
| #define STR_ERR_SERIAL_MISMATCH             "Serial status mismatch" | ||||
| #define STR_BUSY_PROCESSING                 "busy: processing" | ||||
| #define STR_BUSY_PAUSED_FOR_USER            "busy: paused for user" | ||||
| #define STR_BUSY_PAUSED_FOR_INPUT           "busy: paused for input" | ||||
|   | ||||
| @@ -349,6 +349,31 @@ | ||||
|   #define CALL_IF_EXISTS(Return, That, Method, ...) \ | ||||
|     static_cast<Return>(Private::Call_ ## Method(That, ##__VA_ARGS__)) | ||||
|  | ||||
|   // Compile-time string manipulation | ||||
|   namespace CompileTimeString { | ||||
|     // Simple compile-time parser to find the position of the end of a string | ||||
|     constexpr const char* findStringEnd(const char *str) { | ||||
|       return *str ? findStringEnd(str + 1) : str; | ||||
|     } | ||||
|  | ||||
|     // Check whether a string contains a slash | ||||
|     constexpr bool containsSlash(const char *str) { | ||||
|       return *str == '/' ? true : (*str ? containsSlash(str + 1) : false); | ||||
|     } | ||||
|     // Find the last position of the slash | ||||
|     constexpr const char* findLastSlashPos(const char* str) { | ||||
|       return *str == '/' ? (str + 1) : findLastSlashPos(str - 1); | ||||
|     } | ||||
|     // Compile-time evaluation of the last part of a file path | ||||
|     // Typically used to shorten the path to file in compiled strings | ||||
|     // CompileTimeString::baseName(__FILE__) returns "macros.h" and not /path/to/Marlin/src/core/macros.h | ||||
|     constexpr const char* baseName(const char* str) { | ||||
|       return containsSlash(str) ? findLastSlashPos(findStringEnd(str)) : str; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   #define ONLY_FILENAME CompileTimeString::baseName(__FILE__) | ||||
|  | ||||
| #else | ||||
|  | ||||
|   #define MIN_2(a,b)      ((a)<(b)?(a):(b)) | ||||
|   | ||||
| @@ -52,6 +52,10 @@ PGMSTR(SP_X_LBL, " X:"); PGMSTR(SP_Y_LBL, " Y:"); PGMSTR(SP_Z_LBL, " Z:"); PGMST | ||||
|   #endif | ||||
| #endif | ||||
|  | ||||
| #if ENABLED(MEATPACK) | ||||
|   MeatpackSerial<decltype(_SERIAL_IMPL)> mpSerial(false, _SERIAL_IMPL); | ||||
| #endif | ||||
|  | ||||
| void serialprintPGM(PGM_P str) { | ||||
|   while (const char c = pgm_read_byte(str++)) SERIAL_CHAR(c); | ||||
| } | ||||
|   | ||||
| @@ -24,6 +24,10 @@ | ||||
| #include "../inc/MarlinConfig.h" | ||||
| #include "serial_hook.h" | ||||
|  | ||||
| #if ENABLED(MEATPACK) | ||||
|   #include "../feature/meatpack.h" | ||||
| #endif | ||||
|  | ||||
| // Commonly-used strings in serial output | ||||
| extern const char NUL_STR[], SP_P_STR[], SP_T_STR[], | ||||
|                   X_STR[], Y_STR[], Z_STR[], E_STR[], | ||||
| @@ -69,12 +73,19 @@ extern uint8_t marlin_debug_flags; | ||||
|     typedef MultiSerial<decltype(MYSERIAL0), TERN(HAS_ETHERNET, ConditionalSerial<decltype(MYSERIAL1)>, decltype(MYSERIAL1)), 0>      SerialOutputT; | ||||
|   #endif | ||||
|   extern SerialOutputT          multiSerial; | ||||
|   #define SERIAL_IMPL           multiSerial | ||||
|   #define _SERIAL_IMPL          multiSerial | ||||
| #else | ||||
|   #define _PORT_REDIRECT(n,p)   NOOP | ||||
|   #define _PORT_RESTORE(n)      NOOP | ||||
|   #define SERIAL_ASSERT(P)      NOOP | ||||
|   #define SERIAL_IMPL           MYSERIAL0 | ||||
|   #define _SERIAL_IMPL          MYSERIAL0 | ||||
| #endif | ||||
|  | ||||
| #if ENABLED(MEATPACK) | ||||
|   extern MeatpackSerial<decltype(_SERIAL_IMPL)> mpSerial; | ||||
|   #define SERIAL_IMPL          mpSerial | ||||
| #else | ||||
|   #define SERIAL_IMPL          _SERIAL_IMPL | ||||
| #endif | ||||
|  | ||||
| #define SERIAL_OUT(WHAT, V...)  (void)SERIAL_IMPL.WHAT(V) | ||||
| @@ -294,7 +305,7 @@ void serialprintPGM(PGM_P str); | ||||
| #endif | ||||
|  | ||||
| #define SERIAL_ECHOPGM_P(P)         (serialprintPGM(P)) | ||||
| #define SERIAL_ECHOLNPGM_P(P)       (serialprintPGM(P "\n")) | ||||
| #define SERIAL_ECHOLNPGM_P(P)       do{ serialprintPGM(P); SERIAL_EOL(); }while(0) | ||||
|  | ||||
| #define SERIAL_ECHOPGM(S)           (serialprintPGM(PSTR(S))) | ||||
| #define SERIAL_ECHOLNPGM(S)         (serialprintPGM(PSTR(S "\n"))) | ||||
|   | ||||
| @@ -110,7 +110,7 @@ void MeatPack::handle_rx_char_inner(const uint8_t c) { | ||||
|   if (TEST(state, MPConfig_Bit_Active)) {                   // Is MeatPack active? | ||||
|     if (!full_char_count) {                                 // No literal characters to fetch? | ||||
|       uint8_t buf[2] = { 0, 0 }; | ||||
|       register const uint8_t res = unpack_chars(c, buf);    // Decode the byte into one or two characters. | ||||
|       const uint8_t res = unpack_chars(c, buf);             // Decode the byte into one or two characters. | ||||
|       if (res & kFirstCharIsLiteral) {                      // The 1st character couldn't be packed. | ||||
|         ++full_char_count;                                  // So the next stream byte is a full character. | ||||
|         if (res & kSecondCharIsLiteral) ++full_char_count;  // The 2nd character couldn't be packed. Another stream byte is a full character. | ||||
| @@ -147,9 +147,7 @@ void MeatPack::handle_output_char(const uint8_t c) { | ||||
|   #if ENABLED(MP_DEBUG) | ||||
|     if (chars_decoded < 1024) { | ||||
|       ++chars_decoded; | ||||
|       DEBUG_ECHOPGM("RB: "); | ||||
|       MYSERIAL.print((char)c); | ||||
|       DEBUG_EOL(); | ||||
|       DEBUG_ECHOLNPAIR("RB: ", AS_CHAR(c)); | ||||
|     } | ||||
|   #endif | ||||
| } | ||||
| @@ -200,7 +198,7 @@ void MeatPack::handle_rx_char(const uint8_t c, const serial_index_t serial_ind) | ||||
|   } | ||||
|  | ||||
|   if (cmd_is_next) {                      // Were two command bytes received? | ||||
|     PORT_REDIRECT(serial_ind); | ||||
|     PORT_REDIRECT(SERIAL_PORTMASK(serial_ind)); | ||||
|     handle_command((MeatPack_Command)c);  // Then the byte is a MeatPack command | ||||
|     cmd_is_next = false; | ||||
|     return; | ||||
| @@ -219,7 +217,7 @@ uint8_t MeatPack::get_result_char(char* const __restrict out) { | ||||
|   if (char_out_count) { | ||||
|     res = char_out_count; | ||||
|     char_out_count = 0; | ||||
|     for (register uint8_t i = 0; i < res; ++i) | ||||
|     for (uint8_t i = 0; i < res; ++i) | ||||
|       out[i] = (char)char_out_buf[i]; | ||||
|   } | ||||
|   return res; | ||||
|   | ||||
| @@ -49,6 +49,7 @@ | ||||
| #pragma once | ||||
|  | ||||
| #include <stdint.h> | ||||
| #include "../core/serial_hook.h" | ||||
|  | ||||
| /** | ||||
|  * Commands sent to MeatPack to control its behavior. | ||||
| @@ -78,8 +79,6 @@ enum MeatPack_ConfigStateBits : uint8_t { | ||||
| }; | ||||
|  | ||||
| class MeatPack { | ||||
| private: | ||||
|   friend class GCodeQueue; | ||||
|  | ||||
|   // Utility definitions | ||||
|   static const uint8_t kCommandByte         = 0b11111111, | ||||
| @@ -99,6 +98,7 @@ private: | ||||
|                  char_out_count;  // Stores number of characters to be read out. | ||||
|   static uint8_t char_out_buf[2]; // Output buffer for caching up to 2 characters | ||||
|  | ||||
| public: | ||||
|   // Pass in a character rx'd by SD card or serial. Automatically parses command/ctrl sequences, | ||||
|   // and will control state internally. | ||||
|   static void handle_rx_char(const uint8_t c, const serial_index_t serial_ind); | ||||
| @@ -113,7 +113,6 @@ private: | ||||
|  | ||||
|   static void reset_state(); | ||||
|   static void report_state(); | ||||
|   static uint8_t unpacked_char(register const uint8_t in); | ||||
|   static uint8_t unpack_chars(const uint8_t pk, uint8_t* __restrict const chars_out); | ||||
|   static void handle_command(const MeatPack_Command c); | ||||
|   static void handle_output_char(const uint8_t c); | ||||
| @@ -121,3 +120,57 @@ private: | ||||
| }; | ||||
|  | ||||
| extern MeatPack meatpack; | ||||
|  | ||||
| // Implement the MeatPack serial class so it's transparent to rest of the code | ||||
| template <typename SerialT> | ||||
| struct MeatpackSerial : public SerialBase <MeatpackSerial < SerialT >> { | ||||
|   typedef SerialBase< MeatpackSerial<SerialT> > BaseClassT; | ||||
|  | ||||
|   SerialT & out; | ||||
|  | ||||
|   char serialBuffer[2]; | ||||
|   uint8_t charCount; | ||||
|   uint8_t readIndex; | ||||
|  | ||||
|   NO_INLINE size_t write(uint8_t c) { return out.write(c); } | ||||
|   void flush()                      { out.flush();  } | ||||
|   void begin(long br)               { out.begin(br); readIndex = 0; } | ||||
|   void end()                        { out.end(); } | ||||
|  | ||||
|   void msgDone()                    { out.msgDone(); } | ||||
|   // Existing instances implement Arduino's operator bool, so use that if it's available | ||||
|   bool connected()                  { return Private::HasMember_connected<SerialT>::value ? CALL_IF_EXISTS(bool, &out, connected) : (bool)out; } | ||||
|   void flushTX()                    { CALL_IF_EXISTS(void, &out, flushTX); } | ||||
|  | ||||
|   int available(uint8_t index) { | ||||
|     // There is a potential issue here with multiserial, since it'll return its decoded buffer whatever the serial index here. | ||||
|     // So, instead of doing MeatpackSerial<MultiSerial<...>> we should do MultiSerial<MeatpackSerial<...>, MeatpackSerial<...>> | ||||
|     // TODO, let's fix this later on | ||||
|  | ||||
|     if (charCount) return charCount;          // The buffer still has data | ||||
|     if (out.available(index) <= 0) return 0;  // No data to read | ||||
|  | ||||
|     // Don't read in read method, instead do it here, so we can make progress in the read method | ||||
|     const int r = out.read(index); | ||||
|     if (r == -1) return 0;  // This is an error from the underlying serial code | ||||
|     meatpack.handle_rx_char((uint8_t)r, index); | ||||
|     charCount = meatpack.get_result_char(serialBuffer); | ||||
|     readIndex = 0; | ||||
|  | ||||
|     return charCount; | ||||
|   } | ||||
|  | ||||
|   int readImpl(const uint8_t index) { | ||||
|     // Not enough char to make progress? | ||||
|     if (charCount == 0 && available(index) == 0) return -1; | ||||
|  | ||||
|     charCount--; | ||||
|     return serialBuffer[readIndex++]; | ||||
|   } | ||||
|  | ||||
|   int read(uint8_t index) { return readImpl(index); } | ||||
|   int available()         { return available(0); } | ||||
|   int read()              { return readImpl(0); } | ||||
|  | ||||
|   MeatpackSerial(const bool e, SerialT & out) : BaseClassT(e), out(out) {} | ||||
| }; | ||||
|   | ||||
| @@ -51,7 +51,7 @@ | ||||
|  * Also, there are two support functions that can be called from a developer's C code. | ||||
|  * | ||||
|  *    uint16_t check_for_free_memory_corruption(PGM_P const free_memory_start); | ||||
|  *    void M100_dump_routine(PGM_P const title, const char * const start, const char * const end); | ||||
|  *    void M100_dump_routine(PGM_P const title, const char * const start, const uintptr_t size); | ||||
|  * | ||||
|  * Initial version by Roxy-3D | ||||
|  */ | ||||
| @@ -151,7 +151,7 @@ inline int32_t count_test_bytes(const char * const start_free_memory) { | ||||
|    *  the block. If so, it may indicate memory corruption due to a bad pointer. | ||||
|    *  Unexpected bytes are flagged in the right column. | ||||
|    */ | ||||
|   inline void dump_free_memory(char *start_free_memory, char *end_free_memory) { | ||||
|   void dump_free_memory(char *start_free_memory, char *end_free_memory) { | ||||
|     // | ||||
|     // Start and end the dump on a nice 16 byte boundary | ||||
|     // (even though the values are not 16-byte aligned). | ||||
| @@ -182,12 +182,12 @@ inline int32_t count_test_bytes(const char * const start_free_memory) { | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   void M100_dump_routine(PGM_P const title, const char * const start, const char * const end) { | ||||
|     serialprintPGM(title); | ||||
|     SERIAL_EOL(); | ||||
|   void M100_dump_routine(PGM_P const title, const char * const start, const uintptr_t size) { | ||||
|     SERIAL_ECHOLNPGM_P(title); | ||||
|     // | ||||
|     // Round the start and end locations to produce full lines of output | ||||
|     // | ||||
|     const char * const end = start + size - 1; | ||||
|     dump_free_memory( | ||||
|       (char*)(uintptr_t(uint32_t(start) & ~0xFUL)), // Align to 16-byte boundary | ||||
|       (char*)(uintptr_t(uint32_t(end)   |  0xFUL))  // Align end_free_memory to the 15th byte (at or above end_free_memory) | ||||
| @@ -197,27 +197,27 @@ inline int32_t count_test_bytes(const char * const start_free_memory) { | ||||
| #endif // M100_FREE_MEMORY_DUMPER | ||||
|  | ||||
| inline int check_for_free_memory_corruption(PGM_P const title) { | ||||
|   serialprintPGM(title); | ||||
|   SERIAL_ECHOPGM_P(title); | ||||
|  | ||||
|   char *start_free_memory = free_memory_start, *end_free_memory = free_memory_end; | ||||
|   int n = end_free_memory - start_free_memory; | ||||
|  | ||||
|   SERIAL_ECHOPAIR("\nfmc() n=", n); | ||||
|   SERIAL_ECHOPAIR("\nfree_memory_start=", hex_address(free_memory_start)); | ||||
|   SERIAL_ECHOLNPAIR("  end_free_memory=", hex_address(end_free_memory)); | ||||
|   SERIAL_ECHOLNPAIR("\nfmc() n=", n, | ||||
|                     "\nfree_memory_start=", hex_address(free_memory_start), | ||||
|                     "  end=", hex_address(end_free_memory)); | ||||
|  | ||||
|   if (end_free_memory < start_free_memory)  { | ||||
|     SERIAL_ECHOPGM(" end_free_memory < Heap "); | ||||
|     // SET_INPUT_PULLUP(63);           // if the developer has a switch wired up to their controller board | ||||
|     // safe_delay(5);                  // this code can be enabled to pause the display as soon as the | ||||
|     // while ( READ(63))               // malfunction is detected.   It is currently defaulting to a switch | ||||
|     //   idle();                       // being on pin-63 which is unassigend and available on most controller | ||||
|     // safe_delay(20);                 // boards. | ||||
|     // while ( !READ(63)) | ||||
|     //   idle(); | ||||
|     //SET_INPUT_PULLUP(63);           // if the developer has a switch wired up to their controller board | ||||
|     //safe_delay(5);                  // this code can be enabled to pause the display as soon as the | ||||
|     //while ( READ(63))               // malfunction is detected.   It is currently defaulting to a switch | ||||
|     //  idle();                       // being on pin-63 which is unassigend and available on most controller | ||||
|     //safe_delay(20);                 // boards. | ||||
|     //while ( !READ(63)) | ||||
|     //  idle(); | ||||
|     serial_delay(20); | ||||
|     #if ENABLED(M100_FREE_MEMORY_DUMPER) | ||||
|       M100_dump_routine(PSTR("   Memory corruption detected with end_free_memory<Heap\n"), (const char*)0x1B80, (const char*)0x21FF); | ||||
|       M100_dump_routine(PSTR("   Memory corruption detected with end_free_memory<Heap\n"), (const char*)0x1B80, 0x0680); | ||||
|     #endif | ||||
|   } | ||||
|  | ||||
| @@ -227,13 +227,11 @@ inline int check_for_free_memory_corruption(PGM_P const title) { | ||||
|     if (start_free_memory[i] == TEST_BYTE) { | ||||
|       int32_t j = count_test_bytes(start_free_memory + i); | ||||
|       if (j > 8) { | ||||
|         // SERIAL_ECHOPAIR("Found ", j); | ||||
|         // SERIAL_ECHOLNPAIR(" bytes free at ", hex_address(start_free_memory + i)); | ||||
|         //SERIAL_ECHOPAIR("Found ", j); | ||||
|         //SERIAL_ECHOLNPAIR(" bytes free at ", hex_address(start_free_memory + i)); | ||||
|         i += j; | ||||
|         block_cnt++; | ||||
|         SERIAL_ECHOPAIR(" (", block_cnt); | ||||
|         SERIAL_ECHOPAIR(") found=", j); | ||||
|         SERIAL_ECHOLNPGM("   "); | ||||
|         SERIAL_ECHOLNPAIR(" (", block_cnt, ") found=", j); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| @@ -269,8 +267,7 @@ inline void free_memory_pool_report(char * const start_free_memory, const int32_ | ||||
|     if (*addr == TEST_BYTE) { | ||||
|       const int32_t j = count_test_bytes(addr); | ||||
|       if (j > 8) { | ||||
|         SERIAL_ECHOPAIR("Found ", j); | ||||
|         SERIAL_ECHOLNPAIR(" bytes free at ", hex_address(addr)); | ||||
|         SERIAL_ECHOLNPAIR("Found ", j, " bytes free at ", hex_address(addr)); | ||||
|         if (j > max_cnt) { | ||||
|           max_cnt  = j; | ||||
|           max_addr = addr; | ||||
| @@ -280,11 +277,10 @@ inline void free_memory_pool_report(char * const start_free_memory, const int32_ | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|   if (block_cnt > 1) { | ||||
|     SERIAL_ECHOLNPGM("\nMemory Corruption detected in free memory area."); | ||||
|     SERIAL_ECHOPAIR("\nLargest free block is ", max_cnt); | ||||
|     SERIAL_ECHOLNPAIR(" bytes at ", hex_address(max_addr)); | ||||
|   } | ||||
|   if (block_cnt > 1) SERIAL_ECHOLNPAIR( | ||||
|     "\nMemory Corruption detected in free memory area." | ||||
|     "\nLargest free block is ", max_cnt, " bytes at ", hex_address(max_addr) | ||||
|   ); | ||||
|   SERIAL_ECHOLNPAIR("check_for_free_memory_corruption() = ", check_for_free_memory_corruption(PSTR("M100 F "))); | ||||
| } | ||||
|  | ||||
| @@ -294,12 +290,12 @@ inline void free_memory_pool_report(char * const start_free_memory, const int32_ | ||||
|    *  Corrupt <num> locations in the free memory pool and report the corrupt addresses. | ||||
|    *  This is useful to check the correctness of the M100 D and the M100 F commands. | ||||
|    */ | ||||
|   inline void corrupt_free_memory(char *start_free_memory, const uint32_t size) { | ||||
|   inline void corrupt_free_memory(char *start_free_memory, const uintptr_t size) { | ||||
|     start_free_memory += 8; | ||||
|     const uint32_t near_top = top_of_stack() - start_free_memory - 250, // -250 to avoid interrupt activity that's altered the stack. | ||||
|                    j = near_top / (size + 1); | ||||
|  | ||||
|     SERIAL_ECHOLNPGM("Corrupting free memory block.\n"); | ||||
|     SERIAL_ECHOLNPGM("Corrupting free memory block."); | ||||
|     for (uint32_t i = 1; i <= size; i++) { | ||||
|       char * const addr = start_free_memory + i * j; | ||||
|       *addr = i; | ||||
| @@ -322,8 +318,8 @@ inline void init_free_memory(char *start_free_memory, int32_t size) { | ||||
|     return; | ||||
|   } | ||||
|  | ||||
|   start_free_memory += 8;       // move a few bytes away from the heap just because we don't want | ||||
|                   // to be altering memory that close to it. | ||||
|   start_free_memory += 8; // move a few bytes away from the heap just because we | ||||
|                           // don't want to be altering memory that close to it. | ||||
|   memset(start_free_memory, TEST_BYTE, size); | ||||
|  | ||||
|   SERIAL_ECHO(size); | ||||
| @@ -342,16 +338,16 @@ inline void init_free_memory(char *start_free_memory, int32_t size) { | ||||
|  * M100: Free Memory Check | ||||
|  */ | ||||
| void GcodeSuite::M100() { | ||||
|  | ||||
|   char *sp = top_of_stack(); | ||||
|   if (!free_memory_end) free_memory_end = sp - MEMORY_END_CORRECTION; | ||||
|   SERIAL_ECHOPAIR("\nbss_end               : ", hex_address(end_bss)); | ||||
|   if (heaplimit) SERIAL_ECHOPAIR("\n__heaplimit           : ", hex_address(heaplimit)); | ||||
|   SERIAL_ECHOPAIR("\nfree_memory_start     : ", hex_address(free_memory_start)); | ||||
|                   SERIAL_ECHOPAIR("\nbss_end               : ", hex_address(end_bss)); | ||||
|   if (heaplimit)  SERIAL_ECHOPAIR("\n__heaplimit           : ", hex_address(heaplimit)); | ||||
|                   SERIAL_ECHOPAIR("\nfree_memory_start     : ", hex_address(free_memory_start)); | ||||
|   if (stacklimit) SERIAL_ECHOPAIR("\n__stacklimit          : ", hex_address(stacklimit)); | ||||
|   SERIAL_ECHOPAIR("\nfree_memory_end       : ", hex_address(free_memory_end)); | ||||
|   if (MEMORY_END_CORRECTION)  SERIAL_ECHOPAIR("\nMEMORY_END_CORRECTION: ", MEMORY_END_CORRECTION); | ||||
|   SERIAL_ECHOLNPAIR("\nStack Pointer         : ", hex_address(sp)); | ||||
|                   SERIAL_ECHOPAIR("\nfree_memory_end       : ", hex_address(free_memory_end)); | ||||
|   if (MEMORY_END_CORRECTION) | ||||
|                   SERIAL_ECHOPAIR("\nMEMORY_END_CORRECTION : ", MEMORY_END_CORRECTION); | ||||
|                   SERIAL_ECHOLNPAIR("\nStack Pointer       : ", hex_address(sp)); | ||||
|  | ||||
|   // Always init on the first invocation of M100 | ||||
|   static bool m100_not_initialized = true; | ||||
| @@ -369,10 +365,8 @@ void GcodeSuite::M100() { | ||||
|     return free_memory_pool_report(free_memory_start, free_memory_end - free_memory_start); | ||||
|  | ||||
|   #if ENABLED(M100_FREE_MEMORY_CORRUPTOR) | ||||
|  | ||||
|     if (parser.seen('C')) | ||||
|       return corrupt_free_memory(free_memory_start, parser.value_int()); | ||||
|  | ||||
|   #endif | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -260,13 +260,6 @@ void GcodeSuite::dwell(millis_t time) { | ||||
|  | ||||
| #endif // HAS_LEVELING && G29_RETRY_AND_RECOVER | ||||
|  | ||||
| // | ||||
| // Placeholders for non-migrated codes | ||||
| // | ||||
| #if ENABLED(M100_FREE_MEMORY_WATCHER) | ||||
|   extern void M100_dump_routine(PGM_P const title, const char * const start, const char * const end); | ||||
| #endif | ||||
|  | ||||
| /** | ||||
|  * Process the parsed command and dispatch it to its handler | ||||
|  */ | ||||
| @@ -994,30 +987,32 @@ void GcodeSuite::process_parsed_command(const bool no_ok/*=false*/) { | ||||
|   SERIAL_OUT(msgDone); // Call the msgDone serial hook to signal command processing done | ||||
| } | ||||
|  | ||||
| #if ENABLED(M100_FREE_MEMORY_DUMPER) | ||||
|   void M100_dump_routine(PGM_P const title, const char * const start, const uintptr_t size); | ||||
| #endif | ||||
|  | ||||
| /** | ||||
|  * Process a single command and dispatch it to its handler | ||||
|  * This is called from the main loop() | ||||
|  */ | ||||
| void GcodeSuite::process_next_command() { | ||||
|   char * const current_command = queue.command_buffer[queue.index_r]; | ||||
|   GCodeQueue::CommandLine &command = queue.ring_buffer.peek_next_command(); | ||||
|  | ||||
|   PORT_REDIRECT(SERIAL_PORTMASK(queue.port[queue.index_r])); | ||||
|   PORT_REDIRECT(SERIAL_PORTMASK(command.port)); | ||||
|  | ||||
|   #if ENABLED(POWER_LOSS_RECOVERY) | ||||
|     recovery.queue_index_r = queue.index_r; | ||||
|   #endif | ||||
|   TERN_(POWER_LOSS_RECOVERY, recovery.queue_index_r = queue.ring_buffer.index_r); | ||||
|  | ||||
|   if (DEBUGGING(ECHO)) { | ||||
|     SERIAL_ECHO_START(); | ||||
|     SERIAL_ECHOLN(current_command); | ||||
|     SERIAL_ECHOLN(command.buffer); | ||||
|     #if ENABLED(M100_FREE_MEMORY_DUMPER) | ||||
|       SERIAL_ECHOPAIR("slot:", queue.index_r); | ||||
|       M100_dump_routine(PSTR("   Command Queue:"), &queue.command_buffer[0][0], &queue.command_buffer[BUFSIZE - 1][MAX_CMD_SIZE - 1]); | ||||
|       SERIAL_ECHOPAIR("slot:", queue.ring_buffer.index_r); | ||||
|       M100_dump_routine(PSTR("   Command Queue:"), (const char*)&queue.ring_buffer, sizeof(queue.ring_buffer)); | ||||
|     #endif | ||||
|   } | ||||
|  | ||||
|   // Parse the next command in the queue | ||||
|   parser.parse(current_command); | ||||
|   parser.parse(command.buffer); | ||||
|   process_parsed_command(); | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -29,6 +29,6 @@ | ||||
| void GcodeSuite::M110() { | ||||
|  | ||||
|   if (parser.seenval('N')) | ||||
|     queue.last_N[queue.command_port()] = parser.value_long(); | ||||
|     queue.set_current_line_number(parser.value_long()); | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -52,9 +52,7 @@ void GcodeSuite::M118() { | ||||
|     while (*p == ' ') ++p; | ||||
|   } | ||||
|  | ||||
|   #if HAS_MULTI_SERIAL | ||||
|     PORT_REDIRECT(WITHIN(port, 0, NUM_SERIAL) ? (port ? _BV(port - 1) : SERIAL_ALL) : multiSerial.portMask); | ||||
|   #endif | ||||
|   PORT_REDIRECT(WITHIN(port, 0, NUM_SERIAL) ? (port ? SERIAL_PORTMASK(port - 1) : SERIAL_ALL) : multiSerial.portMask); | ||||
|  | ||||
|   if (hasE) SERIAL_ECHO_START(); | ||||
|   if (hasA) SERIAL_ECHOPGM("//"); | ||||
|   | ||||
| @@ -35,6 +35,7 @@ GCodeQueue queue; | ||||
| #include "../module/planner.h" | ||||
| #include "../module/temperature.h" | ||||
| #include "../MarlinCore.h" | ||||
| #include "../core/bug_on.h" | ||||
|  | ||||
| #if ENABLED(PRINTER_EVENT_LEDS) | ||||
|   #include "../feature/leds/printer_event_leds.h" | ||||
| @@ -48,10 +49,6 @@ GCodeQueue queue; | ||||
|   #include "../feature/binary_stream.h" | ||||
| #endif | ||||
|  | ||||
| #if ENABLED(MEATPACK) | ||||
|   #include "../feature/meatpack.h" | ||||
| #endif | ||||
|  | ||||
| #if ENABLED(POWER_LOSS_RECOVERY) | ||||
|   #include "../feature/powerloss.h" | ||||
| #endif | ||||
| @@ -67,44 +64,17 @@ PGMSTR(G28_STR, "G28"); | ||||
|   static millis_t last_command_time = 0; | ||||
| #endif | ||||
|  | ||||
| /** | ||||
|  * GCode line number handling. Hosts may opt to include line numbers when | ||||
|  * sending commands to Marlin, and lines will be checked for sequentiality. | ||||
|  * M110 N<int> sets the current line number. | ||||
|  */ | ||||
| long GCodeQueue::last_N[NUM_SERIAL]; | ||||
| GCodeQueue::SerialState GCodeQueue::serial_state[NUM_SERIAL] = { 0 }; | ||||
| GCodeQueue::RingBuffer GCodeQueue::ring_buffer = { 0 }; | ||||
|  | ||||
| /** | ||||
|  * GCode Command Queue | ||||
|  * A simple ring buffer of BUFSIZE command strings. | ||||
|  * | ||||
|  * Commands are copied into this buffer by the command injectors | ||||
|  * (immediate, serial, sd card) and they are processed sequentially by | ||||
|  * the main loop. The gcode.process_next_command method parses the next | ||||
|  * command and hands off execution to individual handler functions. | ||||
|  */ | ||||
| uint8_t GCodeQueue::length = 0,  // Count of commands in the queue | ||||
|         GCodeQueue::index_r = 0, // Ring buffer read position | ||||
|         GCodeQueue::index_w = 0; // Ring buffer write position | ||||
|  | ||||
| char GCodeQueue::command_buffer[BUFSIZE][MAX_CMD_SIZE]; | ||||
|  | ||||
| /* | ||||
|  * The port that the command was received on | ||||
|  */ | ||||
| #if HAS_MULTI_SERIAL | ||||
|   serial_index_t GCodeQueue::port[BUFSIZE]; | ||||
| #if NO_TIMEOUTS > 0 | ||||
|   static millis_t last_command_time = 0; | ||||
| #endif | ||||
|  | ||||
| /** | ||||
|  * Serial command injection | ||||
|  */ | ||||
|  | ||||
| // Number of characters read in the current line of serial input | ||||
| static int serial_count[NUM_SERIAL] = { 0 }; | ||||
|  | ||||
| bool send_ok[BUFSIZE]; | ||||
|  | ||||
| /** | ||||
|  * Next Injected PROGMEM Command pointer. (nullptr == empty) | ||||
|  * Internal commands are enqueued ahead of serial / SD commands. | ||||
| @@ -116,38 +86,16 @@ PGM_P GCodeQueue::injected_commands_P; // = nullptr | ||||
|  */ | ||||
| char GCodeQueue::injected_commands[64]; // = { 0 } | ||||
|  | ||||
| GCodeQueue::GCodeQueue() { | ||||
|   // Send "ok" after commands by default | ||||
|   LOOP_L_N(i, COUNT(send_ok)) send_ok[i] = true; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Check whether there are any commands yet to be executed | ||||
|  */ | ||||
| bool GCodeQueue::has_commands_queued() { | ||||
|   return queue.length || injected_commands_P || injected_commands[0]; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Clear the Marlin command queue | ||||
|  */ | ||||
| void GCodeQueue::clear() { | ||||
|   index_r = index_w = length = 0; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Once a new command is in the ring buffer, call this to commit it | ||||
|  */ | ||||
| void GCodeQueue::_commit_command(bool say_ok | ||||
| void GCodeQueue::RingBuffer::commit_command(bool skip_ok | ||||
|   #if HAS_MULTI_SERIAL | ||||
|     , serial_index_t serial_ind/*=-1*/ | ||||
|   #endif | ||||
| ) { | ||||
|   send_ok[index_w] = say_ok; | ||||
|   TERN_(HAS_MULTI_SERIAL, port[index_w] = serial_ind); | ||||
|   commands[index_w].skip_ok = skip_ok; | ||||
|   TERN_(HAS_MULTI_SERIAL, commands[index_w].port = serial_ind); | ||||
|   TERN_(POWER_LOSS_RECOVERY, recovery.commit_sdpos(index_w)); | ||||
|   if (++index_w >= BUFSIZE) index_w = 0; | ||||
|   length++; | ||||
|   advance_pos(index_w, 1); | ||||
| } | ||||
|  | ||||
| /** | ||||
| @@ -155,14 +103,14 @@ void GCodeQueue::_commit_command(bool say_ok | ||||
|  * Return true if the command was successfully added. | ||||
|  * Return false for a full buffer, or if the 'command' is a comment. | ||||
|  */ | ||||
| bool GCodeQueue::_enqueue(const char* cmd, bool say_ok/*=false*/ | ||||
| bool GCodeQueue::RingBuffer::enqueue(const char* cmd, bool skip_ok/*=true*/ | ||||
|   #if HAS_MULTI_SERIAL | ||||
|     , serial_index_t serial_ind/*=-1*/ | ||||
|   #endif | ||||
| ) { | ||||
|   if (*cmd == ';' || length >= BUFSIZE) return false; | ||||
|   strcpy(command_buffer[index_w], cmd); | ||||
|   _commit_command(say_ok | ||||
|   strcpy(commands[index_w].buffer, cmd); | ||||
|   commit_command(skip_ok | ||||
|     #if HAS_MULTI_SERIAL | ||||
|       , serial_ind | ||||
|     #endif | ||||
| @@ -175,14 +123,11 @@ bool GCodeQueue::_enqueue(const char* cmd, bool say_ok/*=false*/ | ||||
|  * Return true if the command was consumed | ||||
|  */ | ||||
| bool GCodeQueue::enqueue_one(const char* cmd) { | ||||
|  | ||||
|   //SERIAL_ECHOPGM("enqueue_one(\""); | ||||
|   //SERIAL_ECHO(cmd); | ||||
|   //SERIAL_ECHOPGM("\") \n"); | ||||
|   //SERIAL_ECHOLNPAIR("enqueue_one(\"", cmd, "\")"); | ||||
|  | ||||
|   if (*cmd == 0 || ISEOL(*cmd)) return true; | ||||
|  | ||||
|   if (_enqueue(cmd)) { | ||||
|   if (ring_buffer.enqueue(cmd)) { | ||||
|     SERIAL_ECHO_MSG(STR_ENQUEUEING, cmd, "\""); | ||||
|     return true; | ||||
|   } | ||||
| @@ -260,7 +205,7 @@ bool GCodeQueue::enqueue_one_P(PGM_P const pgcode) { | ||||
|   char cmd[i + 1]; | ||||
|   memcpy_P(cmd, p, i); | ||||
|   cmd[i] = '\0'; | ||||
|   return _enqueue(cmd); | ||||
|   return ring_buffer.enqueue(cmd); | ||||
| } | ||||
|  | ||||
| /** | ||||
| @@ -291,20 +236,21 @@ void GCodeQueue::enqueue_now_P(PGM_P const pgcode) { | ||||
|  *   P<int>  Planner space remaining | ||||
|  *   B<int>  Block queue space remaining | ||||
|  */ | ||||
| void GCodeQueue::ok_to_send() { | ||||
| void GCodeQueue::RingBuffer::ok_to_send() { | ||||
|   #if NO_TIMEOUTS > 0 | ||||
|     // Start counting from the last command's execution | ||||
|     last_command_time = millis(); | ||||
|   #endif | ||||
|   CommandLine &command = commands[index_r]; | ||||
|   #if HAS_MULTI_SERIAL | ||||
|     const serial_index_t serial_ind = command_port(); | ||||
|     const serial_index_t serial_ind = command.port; | ||||
|     if (serial_ind < 0) return; | ||||
|     PORT_REDIRECT(SERIAL_PORTMASK(serial_ind));   // Reply to the serial port that sent the command | ||||
|   #endif | ||||
|   if (!send_ok[index_r]) return; | ||||
|   if (command.skip_ok) return; | ||||
|   SERIAL_ECHOPGM(STR_OK); | ||||
|   #if ENABLED(ADVANCED_OK) | ||||
|     char* p = command_buffer[index_r]; | ||||
|     char* p = command.buffer; | ||||
|     if (*p == 'N') { | ||||
|       SERIAL_CHAR(' ', *p++); | ||||
|       while (NUMERIC_SIGNED(*p)) | ||||
| @@ -321,27 +267,40 @@ void GCodeQueue::ok_to_send() { | ||||
|  * indicate that a command needs to be re-sent. | ||||
|  */ | ||||
| void GCodeQueue::flush_and_request_resend() { | ||||
|   const serial_index_t serial_ind = command_port(); | ||||
|   const serial_index_t serial_ind = ring_buffer.command_port(); | ||||
|   #if HAS_MULTI_SERIAL | ||||
|     if (serial_ind < 0) return;                   // Never mind. Command came from SD or Flash Drive | ||||
|     PORT_REDIRECT(SERIAL_PORTMASK(serial_ind));   // Reply to the serial port that sent the command | ||||
|   #endif | ||||
|   SERIAL_FLUSH(); | ||||
|   SERIAL_ECHOPGM(STR_RESEND); | ||||
|   SERIAL_ECHOLN(last_N[serial_ind] + 1); | ||||
|   ok_to_send(); | ||||
|   SERIAL_ECHOLN(serial_state[serial_ind].last_N + 1); | ||||
| } | ||||
|  | ||||
|  | ||||
| // Multiserial already handle the dispatch to/from multiple port by itself | ||||
| inline bool serial_data_available(uint8_t index = SERIAL_ALL) { | ||||
|   if (index == SERIAL_ALL) { | ||||
|     for (index = 0; index < NUM_SERIAL; index++) { | ||||
|       if (SERIAL_IMPL.available(index) > 0) return true; | ||||
|       const int a = SERIAL_IMPL.available(index); | ||||
|       #if BOTH(RX_BUFFER_MONITOR, RX_BUFFER_SIZE) | ||||
|         if (a > RX_BUFFER_SIZE - 2) { | ||||
|           PORT_REDIRECT(SERIAL_PORTMASK(index)); | ||||
|           SERIAL_ERROR_MSG("RX BUF overflow, increase RX_BUFFER_SIZE: ", a); | ||||
|         } | ||||
|       #endif | ||||
|       if (a > 0) return true; | ||||
|     } | ||||
|     return false; | ||||
|   } | ||||
|   return SERIAL_IMPL.available(index) > 0; | ||||
|   const int a = SERIAL_IMPL.available(index); | ||||
|   #if BOTH(RX_BUFFER_MONITOR, RX_BUFFER_SIZE) | ||||
|     if (a > RX_BUFFER_SIZE - 2) { | ||||
|       PORT_REDIRECT(SERIAL_PORTMASK(index)); | ||||
|       SERIAL_ERROR_MSG("RX BUF overflow, increase RX_BUFFER_SIZE: ", a); | ||||
|     } | ||||
|   #endif | ||||
|  | ||||
|   return a > 0; | ||||
| } | ||||
|  | ||||
| inline int read_serial(const uint8_t index) { return SERIAL_IMPL.read(index); } | ||||
| @@ -349,11 +308,11 @@ inline int read_serial(const uint8_t index) { return SERIAL_IMPL.read(index); } | ||||
| void GCodeQueue::gcode_line_error(PGM_P const err, const serial_index_t serial_ind) { | ||||
|   PORT_REDIRECT(SERIAL_PORTMASK(serial_ind)); // Reply to the serial port that sent the command | ||||
|   SERIAL_ERROR_START(); | ||||
|   serialprintPGM(err); | ||||
|   SERIAL_ECHOLN(last_N[serial_ind]); | ||||
|   while (read_serial(serial_ind) != -1);      // Clear out the RX buffer | ||||
|   SERIAL_ECHOPGM_P(err); | ||||
|   SERIAL_ECHOLN(serial_state[serial_ind].last_N); | ||||
|   while (read_serial(serial_ind) != -1) { /* nada */ } // Clear out the RX buffer. Why don't use flush here ? | ||||
|   flush_and_request_resend(); | ||||
|   serial_count[serial_ind] = 0; | ||||
|   serial_state[serial_ind].count = 0; | ||||
| } | ||||
|  | ||||
| FORCE_INLINE bool is_M29(const char * const cmd) {  // matches "M29" & "M29 ", but not "M290", etc | ||||
| @@ -440,10 +399,6 @@ inline bool process_line_done(uint8_t &sis, char (&buff)[MAX_CMD_SIZE], int &ind | ||||
|  * left on the serial port. | ||||
|  */ | ||||
| void GCodeQueue::get_serial_commands() { | ||||
|   static char serial_line_buffer[NUM_SERIAL][MAX_CMD_SIZE]; | ||||
|  | ||||
|   static uint8_t serial_input_state[NUM_SERIAL] = { PS_NORMAL }; | ||||
|  | ||||
|   #if ENABLED(BINARY_FILE_TRANSFER) | ||||
|     if (card.flag.binary_mode) { | ||||
|       /** | ||||
| @@ -451,7 +406,7 @@ void GCodeQueue::get_serial_commands() { | ||||
|        * receive buffer (which limits the packet size to MAX_CMD_SIZE). | ||||
|        * The receive buffer also limits the packet size for reliable transmission. | ||||
|        */ | ||||
|       binaryStream[card.transfer_port_index].receive(serial_line_buffer[card.transfer_port_index]); | ||||
|       binaryStream[card.transfer_port_index].receive(serial_state[card.transfer_port_index].line_buffer); | ||||
|       return; | ||||
|     } | ||||
|   #endif | ||||
| @@ -460,122 +415,140 @@ void GCodeQueue::get_serial_commands() { | ||||
|   // send "wait" to indicate Marlin is still waiting. | ||||
|   #if NO_TIMEOUTS > 0 | ||||
|     const millis_t ms = millis(); | ||||
|     if (length == 0 && !serial_data_available() && ELAPSED(ms, last_command_time + NO_TIMEOUTS)) { | ||||
|     if (ring_buffer.empty() && !serial_data_available() && ELAPSED(ms, last_command_time + NO_TIMEOUTS)) { | ||||
|       SERIAL_ECHOLNPGM(STR_WAIT); | ||||
|       last_command_time = ms; | ||||
|     } | ||||
|   #endif | ||||
|  | ||||
|   /** | ||||
|    * Loop while serial characters are incoming and the queue is not full | ||||
|    */ | ||||
|   while (length < BUFSIZE && serial_data_available()) { | ||||
|   // Loop while serial characters are incoming and the queue is not full | ||||
|   for (bool hadData = true; hadData;) { | ||||
|     // Unless a serial port has data, this will exit on next iteration | ||||
|     hadData = false; | ||||
|  | ||||
|     LOOP_L_N(p, NUM_SERIAL) { | ||||
|       // Check if the queue is full and exit if it is. | ||||
|       if (ring_buffer.full()) return; | ||||
|  | ||||
|       // No data for this port ? Skip it | ||||
|       if (!serial_data_available(p)) continue; | ||||
|  | ||||
|       // Ok, we have some data to process, let's make progress here | ||||
|       hadData = true; | ||||
|  | ||||
|       const int c = read_serial(p); | ||||
|       if (c < 0) continue; | ||||
|       if (c < 0) { | ||||
|         // This should never happen, let's log it | ||||
|         PORT_REDIRECT(SERIAL_PORTMASK(p));     // Reply to the serial port that sent the command | ||||
|         // Crash here to get more information why it failed | ||||
|         BUG_ON("SP available but read -1"); | ||||
|         SERIAL_ERROR_MSG(STR_ERR_SERIAL_MISMATCH); | ||||
|         SERIAL_FLUSH(); | ||||
|         continue; | ||||
|       } | ||||
|  | ||||
|       #if ENABLED(MEATPACK) | ||||
|         meatpack.handle_rx_char(uint8_t(c), p); | ||||
|         char c_res[2] = { 0, 0 }; | ||||
|         const uint8_t char_count = meatpack.get_result_char(c_res); | ||||
|       #else | ||||
|         constexpr uint8_t char_count = 1; | ||||
|       #endif | ||||
|       const char serial_char = (char)c; | ||||
|       SerialState &serial = serial_state[p]; | ||||
|  | ||||
|       LOOP_L_N(char_index, char_count) { | ||||
|         const char serial_char = TERN(MEATPACK, c_res[char_index], c); | ||||
|       if (ISEOL(serial_char)) { | ||||
|  | ||||
|         if (ISEOL(serial_char)) { | ||||
|         // Reset our state, continue if the line was empty | ||||
|         if (process_line_done(serial.input_state, serial.line_buffer, serial.count)) | ||||
|           continue; | ||||
|  | ||||
|           // Reset our state, continue if the line was empty | ||||
|           if (process_line_done(serial_input_state[p], serial_line_buffer[p], serial_count[p])) | ||||
|             continue; | ||||
|         char* command = serial.line_buffer; | ||||
|  | ||||
|           char* command = serial_line_buffer[p]; | ||||
|         while (*command == ' ') command++;                   // Skip leading spaces | ||||
|         char *npos = (*command == 'N') ? command : nullptr;  // Require the N parameter to start the line | ||||
|  | ||||
|           while (*command == ' ') command++;                   // Skip leading spaces | ||||
|           char *npos = (*command == 'N') ? command : nullptr;  // Require the N parameter to start the line | ||||
|         if (npos) { | ||||
|  | ||||
|           if (npos) { | ||||
|           const bool M110 = !!strstr_P(command, PSTR("M110")); | ||||
|  | ||||
|             const bool M110 = !!strstr_P(command, PSTR("M110")); | ||||
|  | ||||
|             if (M110) { | ||||
|               char* n2pos = strchr(command + 4, 'N'); | ||||
|               if (n2pos) npos = n2pos; | ||||
|             } | ||||
|  | ||||
|             const long gcode_N = strtol(npos + 1, nullptr, 10); | ||||
|  | ||||
|             if (gcode_N != last_N[p] + 1 && !M110) | ||||
|               return gcode_line_error(PSTR(STR_ERR_LINE_NO), p); | ||||
|  | ||||
|             char *apos = strrchr(command, '*'); | ||||
|             if (apos) { | ||||
|               uint8_t checksum = 0, count = uint8_t(apos - command); | ||||
|               while (count) checksum ^= command[--count]; | ||||
|               if (strtol(apos + 1, nullptr, 10) != checksum) | ||||
|                 return gcode_line_error(PSTR(STR_ERR_CHECKSUM_MISMATCH), p); | ||||
|             } | ||||
|             else | ||||
|               return gcode_line_error(PSTR(STR_ERR_NO_CHECKSUM), p); | ||||
|  | ||||
|             last_N[p] = gcode_N; | ||||
|           } | ||||
|           #if ENABLED(SDSUPPORT) | ||||
|             // Pronterface "M29" and "M29 " has no line number | ||||
|             else if (card.flag.saving && !is_M29(command)) | ||||
|               return gcode_line_error(PSTR(STR_ERR_NO_CHECKSUM), p); | ||||
|           #endif | ||||
|  | ||||
|           // | ||||
|           // Movement commands give an alert when the machine is stopped | ||||
|           // | ||||
|  | ||||
|           if (IsStopped()) { | ||||
|             char* gpos = strchr(command, 'G'); | ||||
|             if (gpos) { | ||||
|               switch (strtol(gpos + 1, nullptr, 10)) { | ||||
|                 case 0: case 1: | ||||
|                 #if ENABLED(ARC_SUPPORT) | ||||
|                   case 2: case 3: | ||||
|                 #endif | ||||
|                 #if ENABLED(BEZIER_CURVE_SUPPORT) | ||||
|                   case 5: | ||||
|                 #endif | ||||
|                   PORT_REDIRECT(SERIAL_PORTMASK(p));     // Reply to the serial port that sent the command | ||||
|                   SERIAL_ECHOLNPGM(STR_ERR_STOPPED); | ||||
|                   LCD_MESSAGEPGM(MSG_STOPPED); | ||||
|                   break; | ||||
|               } | ||||
|             } | ||||
|           if (M110) { | ||||
|             char* n2pos = strchr(command + 4, 'N'); | ||||
|             if (n2pos) npos = n2pos; | ||||
|           } | ||||
|  | ||||
|           #if DISABLED(EMERGENCY_PARSER) | ||||
|             // Process critical commands early | ||||
|             if (command[0] == 'M') switch (command[3]) { | ||||
|               case '8': if (command[2] == '0' && command[1] == '1') { wait_for_heatup = false; TERN_(HAS_LCD_MENU, wait_for_user = false); } break; | ||||
|               case '2': if (command[2] == '1' && command[1] == '1') kill(M112_KILL_STR, nullptr, true); break; | ||||
|               case '0': if (command[1] == '4' && command[2] == '1') quickstop_stepper(); break; | ||||
|           const long gcode_N = strtol(npos + 1, nullptr, 10); | ||||
|  | ||||
|           if (gcode_N != serial.last_N + 1 && !M110) { | ||||
|             // In case of error on a serial port, don't prevent other serial port from making progress | ||||
|             gcode_line_error(PSTR(STR_ERR_LINE_NO), p); | ||||
|             break; | ||||
|           } | ||||
|  | ||||
|           char *apos = strrchr(command, '*'); | ||||
|           if (apos) { | ||||
|             uint8_t checksum = 0, count = uint8_t(apos - command); | ||||
|             while (count) checksum ^= command[--count]; | ||||
|             if (strtol(apos + 1, nullptr, 10) != checksum) { | ||||
|               // In case of error on a serial port, don't prevent other serial port from making progress | ||||
|               gcode_line_error(PSTR(STR_ERR_CHECKSUM_MISMATCH), p); | ||||
|               break; | ||||
|             } | ||||
|           #endif | ||||
|           } | ||||
|           else { | ||||
|             // In case of error on a serial port, don't prevent other serial port from making progress | ||||
|             gcode_line_error(PSTR(STR_ERR_NO_CHECKSUM), p); | ||||
|             break; | ||||
|           } | ||||
|  | ||||
|           #if defined(NO_TIMEOUTS) && NO_TIMEOUTS > 0 | ||||
|             last_command_time = ms; | ||||
|           #endif | ||||
|  | ||||
|           // Add the command to the queue | ||||
|           _enqueue(serial_line_buffer[p], true | ||||
|             #if HAS_MULTI_SERIAL | ||||
|               , p | ||||
|             #endif | ||||
|           ); | ||||
|           serial.last_N = gcode_N; | ||||
|         } | ||||
|         else | ||||
|           process_stream_char(serial_char, serial_input_state[p], serial_line_buffer[p], serial_count[p]); | ||||
|         #if ENABLED(SDSUPPORT) | ||||
|           // Pronterface "M29" and "M29 " has no line number | ||||
|           else if (card.flag.saving && !is_M29(command)) { | ||||
|             gcode_line_error(PSTR(STR_ERR_NO_CHECKSUM), p); | ||||
|             break; | ||||
|           } | ||||
|         #endif | ||||
|  | ||||
|       } // char_count loop | ||||
|         // | ||||
|         // Movement commands give an alert when the machine is stopped | ||||
|         // | ||||
|  | ||||
|         if (IsStopped()) { | ||||
|           char* gpos = strchr(command, 'G'); | ||||
|           if (gpos) { | ||||
|             switch (strtol(gpos + 1, nullptr, 10)) { | ||||
|               case 0: case 1: | ||||
|               #if ENABLED(ARC_SUPPORT) | ||||
|                 case 2: case 3: | ||||
|               #endif | ||||
|               #if ENABLED(BEZIER_CURVE_SUPPORT) | ||||
|                 case 5: | ||||
|               #endif | ||||
|                 PORT_REDIRECT(SERIAL_PORTMASK(p));     // Reply to the serial port that sent the command | ||||
|                 SERIAL_ECHOLNPGM(STR_ERR_STOPPED); | ||||
|                 LCD_MESSAGEPGM(MSG_STOPPED); | ||||
|                 break; | ||||
|             } | ||||
|           } | ||||
|         } | ||||
|  | ||||
|         #if DISABLED(EMERGENCY_PARSER) | ||||
|           // Process critical commands early | ||||
|           if (command[0] == 'M') switch (command[3]) { | ||||
|             case '8': if (command[2] == '0' && command[1] == '1') { wait_for_heatup = false; TERN_(HAS_LCD_MENU, wait_for_user = false); } break; | ||||
|             case '2': if (command[2] == '1' && command[1] == '1') kill(M112_KILL_STR, nullptr, true); break; | ||||
|             case '0': if (command[1] == '4' && command[2] == '1') quickstop_stepper(); break; | ||||
|           } | ||||
|         #endif | ||||
|  | ||||
|         #if NO_TIMEOUTS > 0 | ||||
|           last_command_time = ms; | ||||
|         #endif | ||||
|  | ||||
|         // Add the command to the queue | ||||
|         ring_buffer.enqueue(serial.line_buffer, false | ||||
|           #if HAS_MULTI_SERIAL | ||||
|             , p | ||||
|           #endif | ||||
|         ); | ||||
|       } | ||||
|       else | ||||
|         process_stream_char(serial_char, serial.input_state, serial.line_buffer, serial.count); | ||||
|  | ||||
|     } // NUM_SERIAL loop | ||||
|   } // queue has space, serial has data | ||||
| @@ -595,33 +568,35 @@ void GCodeQueue::get_serial_commands() { | ||||
|     if (!IS_SD_PRINTING()) return; | ||||
|  | ||||
|     int sd_count = 0; | ||||
|     while (length < BUFSIZE && !card.eof()) { | ||||
|     while (!ring_buffer.full() && !card.eof()) { | ||||
|       const int16_t n = card.get(); | ||||
|       const bool card_eof = card.eof(); | ||||
|       if (n < 0 && !card_eof) { SERIAL_ERROR_MSG(STR_SD_ERR_READ); continue; } | ||||
|  | ||||
|       CommandLine &command = ring_buffer.commands[ring_buffer.index_w]; | ||||
|       const char sd_char = (char)n; | ||||
|       const bool is_eol = ISEOL(sd_char); | ||||
|       if (is_eol || card_eof) { | ||||
|  | ||||
|  | ||||
|         // Reset stream state, terminate the buffer, and commit a non-empty command | ||||
|         if (!is_eol && sd_count) ++sd_count;          // End of file with no newline | ||||
|         if (!process_line_done(sd_input_state, command_buffer[index_w], sd_count)) { | ||||
|         if (!process_line_done(sd_input_state, command.buffer, sd_count)) { | ||||
|  | ||||
|           // M808 L saves the sdpos of the next line. M808 loops to a new sdpos. | ||||
|           TERN_(GCODE_REPEAT_MARKERS, repeat.early_parse_M808(command_buffer[index_w])); | ||||
|           TERN_(GCODE_REPEAT_MARKERS, repeat.early_parse_M808(command.buffer)); | ||||
|  | ||||
|           // Put the new command into the buffer (no "ok" sent) | ||||
|           _commit_command(false); | ||||
|           ring_buffer.commit_command(true); | ||||
|  | ||||
|           // Prime Power-Loss Recovery for the NEXT _commit_command | ||||
|           // Prime Power-Loss Recovery for the NEXT commit_command | ||||
|           TERN_(POWER_LOSS_RECOVERY, recovery.cmd_sdpos = card.getIndex()); | ||||
|         } | ||||
|  | ||||
|         if (card.eof()) card.fileHasFinished();         // Handle end of file reached | ||||
|       } | ||||
|       else | ||||
|         process_stream_char(sd_char, sd_input_state, command_buffer[index_w], sd_count); | ||||
|         process_stream_char(sd_char, sd_input_state, command.buffer, sd_count); | ||||
|     } | ||||
|   } | ||||
|  | ||||
| @@ -634,6 +609,7 @@ void GCodeQueue::get_serial_commands() { | ||||
|  *  - The SD card file being actively printed | ||||
|  */ | ||||
| void GCodeQueue::get_available_commands() { | ||||
|   if (ring_buffer.full()) return; | ||||
|  | ||||
|   get_serial_commands(); | ||||
|  | ||||
| @@ -649,13 +625,13 @@ void GCodeQueue::advance() { | ||||
|   if (process_injected_command_P() || process_injected_command()) return; | ||||
|  | ||||
|   // Return if the G-code buffer is empty | ||||
|   if (!length) return; | ||||
|   if (ring_buffer.empty()) return; | ||||
|  | ||||
|   #if ENABLED(SDSUPPORT) | ||||
|  | ||||
|     if (card.flag.saving) { | ||||
|       char* command = command_buffer[index_r]; | ||||
|       if (is_M29(command)) { | ||||
|       char * const cmd = ring_buffer.peek_next_command_string(); | ||||
|       if (is_M29(cmd)) { | ||||
|         // M29 closes the file | ||||
|         card.closefile(); | ||||
|         SERIAL_ECHOLNPGM(STR_FILE_SAVED); | ||||
| @@ -673,7 +649,7 @@ void GCodeQueue::advance() { | ||||
|       } | ||||
|       else { | ||||
|         // Write the string from the read buffer to SD | ||||
|         card.write_command(command); | ||||
|         card.write_command(cmd); | ||||
|         if (card.flag.logging) | ||||
|           gcode.process_next_command(); // The card is saving because it's logging | ||||
|         else | ||||
| @@ -690,7 +666,5 @@ void GCodeQueue::advance() { | ||||
|   #endif // SDSUPPORT | ||||
|  | ||||
|   // The queue may be reset by a command handler or by code invoked by idle() within a handler | ||||
|   --length; | ||||
|   if (++index_r >= BUFSIZE) index_r = 0; | ||||
|  | ||||
|   ring_buffer.advance_pos(ring_buffer.index_r, -1); | ||||
| } | ||||
|   | ||||
| @@ -31,41 +31,84 @@ | ||||
| class GCodeQueue { | ||||
| public: | ||||
|   /** | ||||
|    * GCode line number handling. Hosts may include line numbers when sending | ||||
|    * commands to Marlin, and lines will be checked for sequentiality. | ||||
|    * M110 N<int> sets the current line number. | ||||
|    * The buffers per serial port. | ||||
|    */ | ||||
|   struct SerialState { | ||||
|     /** | ||||
|      * GCode line number handling. Hosts may include line numbers when sending | ||||
|      * commands to Marlin, and lines will be checked for sequentiality. | ||||
|      * M110 N<int> sets the current line number. | ||||
|      */ | ||||
|     long last_N; | ||||
|     int count;                        //!< Number of characters read in the current line of serial input | ||||
|     char line_buffer[MAX_CMD_SIZE];   //!< The current line accumulator | ||||
|     uint8_t input_state;              //!< The input state | ||||
|   }; | ||||
|  | ||||
|   static long last_N[NUM_SERIAL]; | ||||
|   static SerialState serial_state[NUM_SERIAL]; //!< Serial states for each serial port | ||||
|  | ||||
|   /** | ||||
|    * GCode Command Queue | ||||
|    * A simple ring buffer of BUFSIZE command strings. | ||||
|    * A simple (circular) ring buffer of BUFSIZE command strings. | ||||
|    * | ||||
|    * Commands are copied into this buffer by the command injectors | ||||
|    * (immediate, serial, sd card) and they are processed sequentially by | ||||
|    * the main loop. The gcode.process_next_command method parses the next | ||||
|    * command and hands off execution to individual handler functions. | ||||
|    */ | ||||
|   static uint8_t length,  // Count of commands in the queue | ||||
|                  index_r; // Ring buffer read position | ||||
|  | ||||
|   static char command_buffer[BUFSIZE][MAX_CMD_SIZE]; | ||||
|   struct CommandLine { | ||||
|     char buffer[MAX_CMD_SIZE];                    //!< The command buffer | ||||
|     bool skip_ok;                                 //!< Skip sending ok when command is processed? | ||||
|     TERN_(HAS_MULTI_SERIAL, serial_index_t port); //!< Serial port the command was received on | ||||
|   }; | ||||
|  | ||||
|   /** | ||||
|    * The port that the command was received on | ||||
|    * A handy ring buffer type | ||||
|    */ | ||||
|   #if HAS_MULTI_SERIAL | ||||
|     static serial_index_t port[BUFSIZE]; | ||||
|   #endif | ||||
|   static inline serial_index_t command_port() { return TERN0(HAS_MULTI_SERIAL, port[index_r]); } | ||||
|   struct RingBuffer { | ||||
|     uint8_t length,                 //!< Number of commands in the queue | ||||
|             index_r,                //!< Ring buffer's read position | ||||
|             index_w;                //!< Ring buffer's write position | ||||
|     CommandLine commands[BUFSIZE];  //!< The ring buffer of commands | ||||
|  | ||||
|   GCodeQueue(); | ||||
|     inline serial_index_t command_port() const { return TERN0(HAS_MULTI_SERIAL, commands[index_r].port); } | ||||
|  | ||||
|     inline void clear() { length = index_r = index_w = 0; } | ||||
|  | ||||
|     void advance_pos(uint8_t &p, const int inc) { if (++p >= BUFSIZE) p = 0; length += inc; } | ||||
|  | ||||
|     void commit_command(bool skip_ok | ||||
|       #if HAS_MULTI_SERIAL | ||||
|         , serial_index_t serial_ind=-1 | ||||
|       #endif | ||||
|     ); | ||||
|  | ||||
|     bool enqueue(const char* cmd, bool skip_ok = true | ||||
|       #if HAS_MULTI_SERIAL | ||||
|         , serial_index_t serial_ind=-1 | ||||
|       #endif | ||||
|     ); | ||||
|  | ||||
|     void ok_to_send(); | ||||
|  | ||||
|     inline bool full(uint8_t cmdCount=1) const { return length > (BUFSIZE - cmdCount); } | ||||
|  | ||||
|     inline bool empty() const { return length == 0; } | ||||
|  | ||||
|     inline CommandLine& peek_next_command() { return commands[index_r]; } | ||||
|  | ||||
|     inline char* peek_next_command_string() { return peek_next_command().buffer; } | ||||
|   }; | ||||
|  | ||||
|   /** | ||||
|    * The ring buffer of commands | ||||
|    */ | ||||
|   static RingBuffer ring_buffer; | ||||
|  | ||||
|   /** | ||||
|    * Clear the Marlin command queue | ||||
|    */ | ||||
|   static void clear(); | ||||
|   static void clear() { ring_buffer.clear(); } | ||||
|  | ||||
|   /** | ||||
|    * Next Injected Command (PROGMEM) pointer. (nullptr == empty) | ||||
| @@ -112,7 +155,7 @@ public: | ||||
|   /** | ||||
|    * Check whether there are any commands yet to be executed | ||||
|    */ | ||||
|   static bool has_commands_queued(); | ||||
|   static bool has_commands_queued() { return ring_buffer.length || injected_commands_P || injected_commands[0]; } | ||||
|  | ||||
|   /** | ||||
|    * Get the next command in the queue, optionally log it to SD, then dispatch it | ||||
| @@ -136,7 +179,7 @@ public: | ||||
|    *   P<int>  Planner space remaining | ||||
|    *   B<int>  Block queue space remaining | ||||
|    */ | ||||
|   static void ok_to_send(); | ||||
|   static inline void ok_to_send() { ring_buffer.ok_to_send(); } | ||||
|  | ||||
|   /** | ||||
|    * Clear the serial line and request a resend of | ||||
| @@ -144,9 +187,12 @@ public: | ||||
|    */ | ||||
|   static void flush_and_request_resend(); | ||||
|  | ||||
| private: | ||||
|   /** | ||||
|    * (Re)Set the current line number for the last received command | ||||
|    */ | ||||
|   static inline void set_current_line_number(long n) { serial_state[ring_buffer.command_port()].last_N = n; } | ||||
|  | ||||
|   static uint8_t index_w;  // Ring buffer write position | ||||
| private: | ||||
|  | ||||
|   static void get_serial_commands(); | ||||
|  | ||||
| @@ -154,18 +200,6 @@ private: | ||||
|     static void get_sdcard_commands(); | ||||
|   #endif | ||||
|  | ||||
|   static void _commit_command(bool say_ok | ||||
|     #if HAS_MULTI_SERIAL | ||||
|       , serial_index_t serial_ind=-1 | ||||
|     #endif | ||||
|   ); | ||||
|  | ||||
|   static bool _enqueue(const char* cmd, bool say_ok=false | ||||
|     #if HAS_MULTI_SERIAL | ||||
|       , serial_index_t serial_ind=-1 | ||||
|     #endif | ||||
|   ); | ||||
|  | ||||
|   // Process the next "immediate" command (PROGMEM) | ||||
|   static bool process_injected_command_P(); | ||||
|  | ||||
| @@ -180,6 +214,7 @@ private: | ||||
|  | ||||
|   static void gcode_line_error(PGM_P const err, const serial_index_t serial_ind); | ||||
|  | ||||
|   friend class GcodeSuite; | ||||
| }; | ||||
|  | ||||
| extern GCodeQueue queue; | ||||
|   | ||||
| @@ -162,7 +162,7 @@ static void lv_kb_event_cb(lv_obj_t *kb, lv_event_t event) { | ||||
|           draw_return_ui(); | ||||
|           break; | ||||
|         case GCodeCommand: | ||||
|           if (queue.length <= (BUFSIZE - 3)) { | ||||
|           if (!queue.ring_buffer.full(3)) { | ||||
|             // Hook anything that goes to the serial port | ||||
|             MYSERIAL0.setHook(lv_serial_capt_hook, lv_eom_hook, 0); | ||||
|             queue.enqueue_one_now(ret_ta_txt); | ||||
|   | ||||
| @@ -46,7 +46,7 @@ static void event_handler(lv_obj_t *obj, lv_event_t event) { | ||||
|  | ||||
|   switch (obj->mks_obj_id) { | ||||
|     case ID_M_POINT1 ... ID_M_POINT5: | ||||
|       if (queue.length == 0) { | ||||
|       if (queue.ring_buffer.empty()) { | ||||
|         if (uiCfg.leveling_first_time) { | ||||
|           uiCfg.leveling_first_time = false; | ||||
|           queue.inject_P(G28_STR); | ||||
|   | ||||
| @@ -54,7 +54,7 @@ enum { | ||||
| static void event_handler(lv_obj_t *obj, lv_event_t event) { | ||||
|   char str_1[16]; | ||||
|   if (event != LV_EVENT_RELEASED) return; | ||||
|   if (queue.length <= (BUFSIZE - 3)) { | ||||
|   if (!queue.ring_buffer.full(3)) { | ||||
|     bool do_inject = true; | ||||
|     float dist = uiCfg.move_dist; | ||||
|     switch (obj->mks_obj_id) { | ||||
|   | ||||
| @@ -1613,7 +1613,7 @@ void wifi_rcv_handle() { | ||||
|       if (wifiTransError.flag != 0x1) WIFI_IO1_RESET(); | ||||
|       getDataF = 1; | ||||
|     } | ||||
|     if (need_ok_later &&  (queue.length < BUFSIZE)) { | ||||
|     if (need_ok_later && !queue.ring_buffer.full()) { | ||||
|       need_ok_later = false; | ||||
|       send_to_wifi((uint8_t *)"ok\r\n", strlen("ok\r\n")); | ||||
|     } | ||||
| @@ -1772,7 +1772,7 @@ void get_wifi_commands() { | ||||
|   static int wifi_read_count = 0; | ||||
|  | ||||
|   if (espGcodeFifo.wait_tick > 5) { | ||||
|     while ((queue.length < BUFSIZE) && (espGcodeFifo.r != espGcodeFifo.w)) { | ||||
|     while (!queue.ring_buffer.full() && (espGcodeFifo.r != espGcodeFifo.w)) { | ||||
|  | ||||
|       espGcodeFifo.wait_tick = 0; | ||||
|  | ||||
|   | ||||
		Reference in New Issue
	
	Block a user