Make CardReader class static (#12451)
* Make CardReader a static class * Make CardReader flags into bitfields
This commit is contained in:
		| @@ -19,7 +19,7 @@ void sd_mmc_spi_mem_init(void) { | |||||||
| } | } | ||||||
|  |  | ||||||
| Ctrl_status sd_mmc_spi_test_unit_ready(void) { | Ctrl_status sd_mmc_spi_test_unit_ready(void) { | ||||||
|   if (!IS_SD_INSERTED() || IS_SD_PRINTING() || IS_SD_FILE_OPEN() || !card.cardOK) |   if (!IS_SD_INSERTED() || IS_SD_PRINTING() || IS_SD_FILE_OPEN() || !card.flag.cardOK) | ||||||
|     return CTRL_NO_PRESENT; |     return CTRL_NO_PRESENT; | ||||||
|   return CTRL_GOOD; |   return CTRL_GOOD; | ||||||
| } | } | ||||||
| @@ -27,7 +27,7 @@ Ctrl_status sd_mmc_spi_test_unit_ready(void) { | |||||||
| // NOTE: This function is defined as returning the address of the last block | // NOTE: This function is defined as returning the address of the last block | ||||||
| // in the card, which is cardSize() - 1 | // in the card, which is cardSize() - 1 | ||||||
| Ctrl_status sd_mmc_spi_read_capacity(uint32_t *nb_sector) { | Ctrl_status sd_mmc_spi_read_capacity(uint32_t *nb_sector) { | ||||||
|   if (!IS_SD_INSERTED() || IS_SD_PRINTING() || IS_SD_FILE_OPEN() || !card.cardOK) |   if (!IS_SD_INSERTED() || IS_SD_PRINTING() || IS_SD_FILE_OPEN() || !card.flag.cardOK) | ||||||
|     return CTRL_NO_PRESENT; |     return CTRL_NO_PRESENT; | ||||||
|   *nb_sector = card.getSd2Card().cardSize() - 1; |   *nb_sector = card.getSd2Card().cardSize() - 1; | ||||||
|   return CTRL_GOOD; |   return CTRL_GOOD; | ||||||
| @@ -42,7 +42,7 @@ bool sd_mmc_spi_wr_protect(void) { | |||||||
| } | } | ||||||
|  |  | ||||||
| bool sd_mmc_spi_removal(void) { | bool sd_mmc_spi_removal(void) { | ||||||
|   if (!IS_SD_INSERTED() || IS_SD_PRINTING() || IS_SD_FILE_OPEN() || !card.cardOK) |   if (!IS_SD_INSERTED() || IS_SD_PRINTING() || IS_SD_FILE_OPEN() || !card.flag.cardOK) | ||||||
|     return true; |     return true; | ||||||
|   return false; |   return false; | ||||||
| } | } | ||||||
| @@ -61,7 +61,7 @@ uint8_t sector_buf[SD_MMC_BLOCK_SIZE]; | |||||||
| // #define DEBUG_MMC | // #define DEBUG_MMC | ||||||
|  |  | ||||||
| Ctrl_status sd_mmc_spi_usb_read_10(uint32_t addr, uint16_t nb_sector) { | Ctrl_status sd_mmc_spi_usb_read_10(uint32_t addr, uint16_t nb_sector) { | ||||||
|   if (!IS_SD_INSERTED() || IS_SD_PRINTING() || IS_SD_FILE_OPEN() || !card.cardOK) |   if (!IS_SD_INSERTED() || IS_SD_PRINTING() || IS_SD_FILE_OPEN() || !card.flag.cardOK) | ||||||
|     return CTRL_NO_PRESENT; |     return CTRL_NO_PRESENT; | ||||||
|  |  | ||||||
|   #ifdef DEBUG_MMC |   #ifdef DEBUG_MMC | ||||||
| @@ -95,7 +95,7 @@ Ctrl_status sd_mmc_spi_usb_read_10(uint32_t addr, uint16_t nb_sector) { | |||||||
| } | } | ||||||
|  |  | ||||||
| Ctrl_status sd_mmc_spi_usb_write_10(uint32_t addr, uint16_t nb_sector) { | Ctrl_status sd_mmc_spi_usb_write_10(uint32_t addr, uint16_t nb_sector) { | ||||||
|   if (!IS_SD_INSERTED() || IS_SD_PRINTING() || IS_SD_FILE_OPEN() || !card.cardOK) |   if (!IS_SD_INSERTED() || IS_SD_PRINTING() || IS_SD_FILE_OPEN() || !card.flag.cardOK) | ||||||
|     return CTRL_NO_PRESENT; |     return CTRL_NO_PRESENT; | ||||||
|  |  | ||||||
|   #ifdef DEBUG_MMC |   #ifdef DEBUG_MMC | ||||||
|   | |||||||
| @@ -106,7 +106,7 @@ void HAL_idletask(void) { | |||||||
|     // the disk if Marlin has it mounted. Unfortuately there is currently no way |     // the disk if Marlin has it mounted. Unfortuately there is currently no way | ||||||
|     // to unmount the disk from the LCD menu. |     // to unmount the disk from the LCD menu. | ||||||
|     // if (IS_SD_PRINTING() || IS_SD_FILE_OPEN()) |     // if (IS_SD_PRINTING() || IS_SD_FILE_OPEN()) | ||||||
|     if (card.cardOK) |     if (card.flag.cardOK) | ||||||
|       MSC_Aquire_Lock(); |       MSC_Aquire_Lock(); | ||||||
|     else |     else | ||||||
|       MSC_Release_Lock(); |       MSC_Release_Lock(); | ||||||
|   | |||||||
| @@ -41,7 +41,7 @@ char HAL_STM32F1_eeprom_content[HAL_STM32F1_EEPROM_SIZE]; | |||||||
| char eeprom_filename[] = "eeprom.dat"; | char eeprom_filename[] = "eeprom.dat"; | ||||||
|  |  | ||||||
| bool PersistentStore::access_start() { | bool PersistentStore::access_start() { | ||||||
|   if (!card.cardOK) return false; |   if (!card.flag.cardOK) return false; | ||||||
|   int16_t bytes_read = 0; |   int16_t bytes_read = 0; | ||||||
|   constexpr char eeprom_zero = 0xFF; |   constexpr char eeprom_zero = 0xFF; | ||||||
|   card.openFile(eeprom_filename, true); |   card.openFile(eeprom_filename, true); | ||||||
| @@ -54,7 +54,7 @@ bool PersistentStore::access_start() { | |||||||
| } | } | ||||||
|  |  | ||||||
| bool PersistentStore::access_finish() { | bool PersistentStore::access_finish() { | ||||||
|   if (!card.cardOK) return false; |   if (!card.flag.cardOK) return false; | ||||||
|   card.openFile(eeprom_filename, true); |   card.openFile(eeprom_filename, true); | ||||||
|   int16_t bytes_written = card.write(HAL_STM32F1_eeprom_content, HAL_STM32F1_EEPROM_SIZE); |   int16_t bytes_written = card.write(HAL_STM32F1_eeprom_content, HAL_STM32F1_EEPROM_SIZE); | ||||||
|   card.closefile(); |   card.closefile(); | ||||||
|   | |||||||
| @@ -967,7 +967,7 @@ void loop() { | |||||||
|     #if ENABLED(SDSUPPORT) |     #if ENABLED(SDSUPPORT) | ||||||
|       card.checkautostart(); |       card.checkautostart(); | ||||||
|  |  | ||||||
|       if (card.abort_sd_printing) { |       if (card.flag.abort_sd_printing) { | ||||||
|         card.stopSDPrint( |         card.stopSDPrint( | ||||||
|           #if SD_RESORT |           #if SD_RESORT | ||||||
|             true |             true | ||||||
|   | |||||||
| @@ -84,8 +84,8 @@ void PrintJobRecovery::changed() { | |||||||
|  */ |  */ | ||||||
| void PrintJobRecovery::check() { | void PrintJobRecovery::check() { | ||||||
|   if (enabled) { |   if (enabled) { | ||||||
|     if (!card.cardOK) card.initsd(); |     if (!card.flag.cardOK) card.initsd(); | ||||||
|     if (card.cardOK) { |     if (card.flag.cardOK) { | ||||||
|       load(); |       load(); | ||||||
|       if (!valid()) return purge(); |       if (!valid()) return purge(); | ||||||
|       enqueue_and_echo_commands_P(PSTR("M1000 S")); |       enqueue_and_echo_commands_P(PSTR("M1000 S")); | ||||||
|   | |||||||
| @@ -68,7 +68,7 @@ typedef struct { | |||||||
|   char command_queue[BUFSIZE][MAX_CMD_SIZE]; |   char command_queue[BUFSIZE][MAX_CMD_SIZE]; | ||||||
|  |  | ||||||
|   // SD Filename and position |   // SD Filename and position | ||||||
|   char sd_filename[MAXPATHNAMELENGTH]; |   char sd_filename[MAXPATHNAMELENGTH + 1]; | ||||||
|   uint32_t sdpos; |   uint32_t sdpos; | ||||||
|  |  | ||||||
|   // Job elapsed time |   // Job elapsed time | ||||||
|   | |||||||
| @@ -503,7 +503,7 @@ static int read_serial(const uint8_t index) { | |||||||
|             break; |             break; | ||||||
|           case StreamState::STREAM_COMPLETE: |           case StreamState::STREAM_COMPLETE: | ||||||
|             stream_state = StreamState::STREAM_RESET; |             stream_state = StreamState::STREAM_RESET; | ||||||
|             card.binary_mode = false; |             card.flag.binary_mode = false; | ||||||
|             card.closefile(); |             card.closefile(); | ||||||
|             CARD_ECHO_P("echo: "); |             CARD_ECHO_P("echo: "); | ||||||
|             CARD_ECHO_P(card.filename); |             CARD_ECHO_P(card.filename); | ||||||
| @@ -514,7 +514,7 @@ static int read_serial(const uint8_t index) { | |||||||
|             return; |             return; | ||||||
|           case StreamState::STREAM_FAILED: |           case StreamState::STREAM_FAILED: | ||||||
|             stream_state = StreamState::STREAM_RESET; |             stream_state = StreamState::STREAM_RESET; | ||||||
|             card.binary_mode = false; |             card.flag.binary_mode = false; | ||||||
|             card.closefile(); |             card.closefile(); | ||||||
|             card.removeFile(card.filename); |             card.removeFile(card.filename); | ||||||
|             CARD_ECHOLN_P("echo: File transfer failed"); |             CARD_ECHOLN_P("echo: File transfer failed"); | ||||||
| @@ -549,7 +549,7 @@ inline void get_serial_commands() { | |||||||
|             ; |             ; | ||||||
|  |  | ||||||
|   #if ENABLED(FAST_FILE_TRANSFER) |   #if ENABLED(FAST_FILE_TRANSFER) | ||||||
|     if (card.saving && card.binary_mode) { |     if (card.flag.saving && card.flag.binary_mode) { | ||||||
|       /** |       /** | ||||||
|        * For binary stream file transfer, use serial_line_buffer as the working |        * For binary stream file transfer, use serial_line_buffer as the working | ||||||
|        * receive buffer (which limits the packet size to MAX_CMD_SIZE). |        * receive buffer (which limits the packet size to MAX_CMD_SIZE). | ||||||
| @@ -630,7 +630,7 @@ inline void get_serial_commands() { | |||||||
|           gcode_LastN = gcode_N; |           gcode_LastN = gcode_N; | ||||||
|         } |         } | ||||||
|         #if ENABLED(SDSUPPORT) |         #if ENABLED(SDSUPPORT) | ||||||
|           else if (card.saving && strcmp(command, "M29") != 0) // No line number with M29 in Pronterface |           else if (card.flag.saving && strcmp(command, "M29") != 0) // No line number with M29 in Pronterface | ||||||
|             return gcode_line_error(PSTR(MSG_ERR_NO_CHECKSUM), i); |             return gcode_line_error(PSTR(MSG_ERR_NO_CHECKSUM), i); | ||||||
|         #endif |         #endif | ||||||
|  |  | ||||||
| @@ -838,7 +838,7 @@ void advance_command_queue() { | |||||||
|  |  | ||||||
|   #if ENABLED(SDSUPPORT) |   #if ENABLED(SDSUPPORT) | ||||||
|  |  | ||||||
|     if (card.saving) { |     if (card.flag.saving) { | ||||||
|       char* command = command_queue[cmd_queue_index_r]; |       char* command = command_queue[cmd_queue_index_r]; | ||||||
|       if (strstr_P(command, PSTR("M29"))) { |       if (strstr_P(command, PSTR("M29"))) { | ||||||
|         // M29 closes the file |         // M29 closes the file | ||||||
| @@ -860,7 +860,7 @@ void advance_command_queue() { | |||||||
|       else { |       else { | ||||||
|         // Write the string from the read buffer to SD |         // Write the string from the read buffer to SD | ||||||
|         card.write_command(command); |         card.write_command(command); | ||||||
|         if (card.logging) |         if (card.flag.logging) | ||||||
|           gcode.process_next_command(); // The card is saving because it's logging |           gcode.process_next_command(); // The card is saving because it's logging | ||||||
|         else |         else | ||||||
|           ok_to_send(); |           ok_to_send(); | ||||||
|   | |||||||
| @@ -118,7 +118,7 @@ void GcodeSuite::M25() { | |||||||
|  * M26: Set SD Card file index |  * M26: Set SD Card file index | ||||||
|  */ |  */ | ||||||
| void GcodeSuite::M26() { | void GcodeSuite::M26() { | ||||||
|   if (card.cardOK && parser.seenval('S')) |   if (card.flag.cardOK && parser.seenval('S')) | ||||||
|     card.setIndex(parser.value_long()); |     card.setIndex(parser.value_long()); | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -178,7 +178,7 @@ void GcodeSuite::M28() { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     // Binary transfer mode |     // Binary transfer mode | ||||||
|     if ((card.binary_mode = binary_mode)) { |     if ((card.flag.binary_mode = binary_mode)) { | ||||||
|       SERIAL_ECHO_START_P(port); |       SERIAL_ECHO_START_P(port); | ||||||
|       SERIAL_ECHO_P(port, " preparing to receive: "); |       SERIAL_ECHO_P(port, " preparing to receive: "); | ||||||
|       SERIAL_ECHOLN_P(port, p); |       SERIAL_ECHOLN_P(port, p); | ||||||
| @@ -202,14 +202,14 @@ void GcodeSuite::M28() { | |||||||
|  * Processed in write to file routine |  * Processed in write to file routine | ||||||
|  */ |  */ | ||||||
| void GcodeSuite::M29() { | void GcodeSuite::M29() { | ||||||
|   // card.saving = false; |   // card.flag.saving = false; | ||||||
| } | } | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * M30 <filename>: Delete SD Card file |  * M30 <filename>: Delete SD Card file | ||||||
|  */ |  */ | ||||||
| void GcodeSuite::M30() { | void GcodeSuite::M30() { | ||||||
|   if (card.cardOK) { |   if (card.flag.cardOK) { | ||||||
|     card.closefile(); |     card.closefile(); | ||||||
|     card.removeFile(parser.string_arg); |     card.removeFile(parser.string_arg); | ||||||
|   } |   } | ||||||
| @@ -228,7 +228,7 @@ void GcodeSuite::M30() { | |||||||
| void GcodeSuite::M32() { | void GcodeSuite::M32() { | ||||||
|   if (IS_SD_PRINTING()) planner.synchronize(); |   if (IS_SD_PRINTING()) planner.synchronize(); | ||||||
|  |  | ||||||
|   if (card.cardOK) { |   if (card.flag.cardOK) { | ||||||
|     const bool call_procedure = parser.boolval('P'); |     const bool call_procedure = parser.boolval('P'); | ||||||
|  |  | ||||||
|     card.openFile(parser.string_arg, true, call_procedure); |     card.openFile(parser.string_arg, true, call_procedure); | ||||||
| @@ -286,7 +286,7 @@ void GcodeSuite::M32() { | |||||||
|  * M524: Abort the current SD print job (started with M24) |  * M524: Abort the current SD print job (started with M24) | ||||||
|  */ |  */ | ||||||
| void GcodeSuite::M524() { | void GcodeSuite::M524() { | ||||||
|   if (IS_SD_PRINTING()) card.abort_sd_printing = true; |   if (IS_SD_PRINTING()) card.flag.abort_sd_printing = true; | ||||||
| } | } | ||||||
|  |  | ||||||
| /** | /** | ||||||
|   | |||||||
| @@ -575,7 +575,7 @@ namespace ExtUI { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   bool isPrintingFromMedia() { |   bool isPrintingFromMedia() { | ||||||
|     return IFSD(card.cardOK && card.isFileOpen(), false); |     return IFSD(card.flag.cardOK && card.isFileOpen(), false); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   bool isPrinting() { |   bool isPrinting() { | ||||||
| @@ -583,7 +583,7 @@ namespace ExtUI { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   bool isMediaInserted() { |   bool isMediaInserted() { | ||||||
|     return IFSD(IS_SD_INSERTED() && card.cardOK, false); |     return IFSD(IS_SD_INSERTED() && card.flag.cardOK, false); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   void pausePrint() { |   void pausePrint() { | ||||||
| @@ -612,7 +612,7 @@ namespace ExtUI { | |||||||
|   void stopPrint() { |   void stopPrint() { | ||||||
|     #if ENABLED(SDSUPPORT) |     #if ENABLED(SDSUPPORT) | ||||||
|       wait_for_heatup = wait_for_user = false; |       wait_for_heatup = wait_for_user = false; | ||||||
|       card.abort_sd_printing = true; |       card.flag.abort_sd_printing = true; | ||||||
|       ExtUI::onStatusChanged(PSTR(MSG_PRINT_ABORTED)); |       ExtUI::onStatusChanged(PSTR(MSG_PRINT_ABORTED)); | ||||||
|     #endif |     #endif | ||||||
|   } |   } | ||||||
| @@ -648,7 +648,7 @@ namespace ExtUI { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   bool FileList::isDir() { |   bool FileList::isDir() { | ||||||
|     return IFSD(card.filenameIsDir, false); |     return IFSD(card.flag.filenameIsDir, false); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   uint16_t FileList::count() { |   uint16_t FileList::count() { | ||||||
| @@ -697,13 +697,13 @@ void MarlinUI::update() { | |||||||
|       last_sd_status = sd_status; |       last_sd_status = sd_status; | ||||||
|       if (sd_status) { |       if (sd_status) { | ||||||
|         card.initsd(); |         card.initsd(); | ||||||
|         if (card.cardOK) |         if (card.flag.cardOK) | ||||||
|           ExtUI::onMediaInserted(); |           ExtUI::onMediaInserted(); | ||||||
|         else |         else | ||||||
|           ExtUI::onMediaError(); |           ExtUI::onMediaError(); | ||||||
|       } |       } | ||||||
|       else { |       else { | ||||||
|         const bool ok = card.cardOK; |         const bool ok = card.flag.cardOK; | ||||||
|         card.release(); |         card.release(); | ||||||
|         if (ok) ExtUI::onMediaRemoved(); |         if (ok) ExtUI::onMediaRemoved(); | ||||||
|       } |       } | ||||||
|   | |||||||
| @@ -278,7 +278,7 @@ void process_lcd_p_command(const char* command) { | |||||||
|         // There may be a difference in how V1 and V2 LCDs handle subdirectory |         // There may be a difference in how V1 and V2 LCDs handle subdirectory | ||||||
|         // prints. Investigate more. This matches the V1 motion controller actions |         // prints. Investigate more. This matches the V1 motion controller actions | ||||||
|         // but the V2 LCD switches to "print" mode on {SYS:DIR} response. |         // but the V2 LCD switches to "print" mode on {SYS:DIR} response. | ||||||
|         if (card.filenameIsDir) { |         if (card.flag.filenameIsDir) { | ||||||
|           card.chdir(card.filename); |           card.chdir(card.filename); | ||||||
|           write_to_lcd_P(PSTR("{SYS:DIR}")); |           write_to_lcd_P(PSTR("{SYS:DIR}")); | ||||||
|         } |         } | ||||||
| @@ -330,7 +330,7 @@ void process_lcd_s_command(const char* command) { | |||||||
|  |  | ||||||
|     case 'L': { |     case 'L': { | ||||||
|       #if ENABLED(SDSUPPORT) |       #if ENABLED(SDSUPPORT) | ||||||
|         if (!card.cardOK) card.initsd(); |         if (!card.flag.cardOK) card.initsd(); | ||||||
|  |  | ||||||
|         // A more efficient way to do this would be to |         // A more efficient way to do this would be to | ||||||
|         // implement a callback in the ls_SerialPrint code, but |         // implement a callback in the ls_SerialPrint code, but | ||||||
| @@ -342,7 +342,7 @@ void process_lcd_s_command(const char* command) { | |||||||
|         uint16_t file_count = card.get_num_Files(); |         uint16_t file_count = card.get_num_Files(); | ||||||
|         for (uint16_t i = 0; i < file_count; i++) { |         for (uint16_t i = 0; i < file_count; i++) { | ||||||
|           card.getfilename(i); |           card.getfilename(i); | ||||||
|           sprintf_P(message_buffer, card.filenameIsDir ? PSTR("{DIR:%s}") : PSTR("{FILE:%s}"), card.longest_filename()); |           sprintf_P(message_buffer, card.flag.filenameIsDir ? PSTR("{DIR:%s}") : PSTR("{FILE:%s}"), card.longest_filename()); | ||||||
|           write_to_lcd(message_buffer); |           write_to_lcd(message_buffer); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -58,7 +58,7 @@ | |||||||
|  |  | ||||||
|   void lcd_sdcard_stop() { |   void lcd_sdcard_stop() { | ||||||
|     wait_for_heatup = wait_for_user = false; |     wait_for_heatup = wait_for_user = false; | ||||||
|     card.abort_sd_printing = true; |     card.flag.abort_sd_printing = true; | ||||||
|     ui.setstatusPGM(PSTR(MSG_PRINT_ABORTED), -1); |     ui.setstatusPGM(PSTR(MSG_PRINT_ABORTED), -1); | ||||||
|     ui.return_to_status(); |     ui.return_to_status(); | ||||||
|   } |   } | ||||||
| @@ -86,7 +86,7 @@ void menu_main() { | |||||||
|   MENU_BACK(MSG_WATCH); |   MENU_BACK(MSG_WATCH); | ||||||
|  |  | ||||||
|   #if ENABLED(SDSUPPORT) |   #if ENABLED(SDSUPPORT) | ||||||
|     if (card.cardOK) { |     if (card.flag.cardOK) { | ||||||
|       if (card.isFileOpen()) { |       if (card.isFileOpen()) { | ||||||
|         if (IS_SD_PRINTING()) |         if (IS_SD_PRINTING()) | ||||||
|           MENU_ITEM(function, MSG_PAUSE_PRINT, lcd_sdcard_pause); |           MENU_ITEM(function, MSG_PAUSE_PRINT, lcd_sdcard_pause); | ||||||
|   | |||||||
| @@ -125,7 +125,7 @@ void menu_sdcard() { | |||||||
|  |  | ||||||
|       card.getfilename_sorted(nr); |       card.getfilename_sorted(nr); | ||||||
|  |  | ||||||
|       if (card.filenameIsDir) |       if (card.flag.filenameIsDir) | ||||||
|         MENU_ITEM(sdfolder, MSG_CARD_MENU, card); |         MENU_ITEM(sdfolder, MSG_CARD_MENU, card); | ||||||
|       else |       else | ||||||
|         MENU_ITEM(sdfile, MSG_CARD_MENU, card); |         MENU_ITEM(sdfile, MSG_CARD_MENU, card); | ||||||
|   | |||||||
| @@ -106,7 +106,7 @@ | |||||||
|  * Defines for 8.3 and long (vfat) filenames |  * Defines for 8.3 and long (vfat) filenames | ||||||
|  */ |  */ | ||||||
|  |  | ||||||
| #define FILENAME_LENGTH 13 // Number of UTF-16 characters per entry | #define FILENAME_LENGTH 12 // Number of UTF-16 characters per entry | ||||||
|  |  | ||||||
| // Total bytes needed to store a single long filename | // Total bytes needed to store a single long filename | ||||||
| #define LONG_FILENAME_LENGTH (FILENAME_LENGTH * MAX_VFAT_ENTRIES + 1) | #define LONG_FILENAME_LENGTH ((FILENAME_LENGTH) * (MAX_VFAT_ENTRIES)) | ||||||
|   | |||||||
| @@ -45,7 +45,83 @@ | |||||||
|   #include "../feature/pause.h" |   #include "../feature/pause.h" | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
| #include <ctype.h> | // public: | ||||||
|  |  | ||||||
|  | card_flags_t CardReader::flag; | ||||||
|  | char CardReader::filename[FILENAME_LENGTH + 1], CardReader::longFilename[LONG_FILENAME_LENGTH + 1]; | ||||||
|  | int8_t CardReader::autostart_index; | ||||||
|  |  | ||||||
|  | #if ENABLED(FAST_FILE_TRANSFER) | ||||||
|  |   #if NUM_SERIAL > 1 | ||||||
|  |     uint8_t CardReader::transfer_port; | ||||||
|  |   #endif | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | // private: | ||||||
|  |  | ||||||
|  | SdFile CardReader::root, CardReader::workDir, CardReader::workDirParents[MAX_DIR_DEPTH]; | ||||||
|  | uint8_t CardReader::workDirDepth; | ||||||
|  |  | ||||||
|  | #if ENABLED(SDCARD_SORT_ALPHA) | ||||||
|  |   uint16_t CardReader::sort_count; | ||||||
|  |   #if ENABLED(SDSORT_GCODE) | ||||||
|  |     bool CardReader::sort_alpha; | ||||||
|  |     int CardReader::sort_folders; | ||||||
|  |     //bool CardReader::sort_reverse; | ||||||
|  |   #endif | ||||||
|  |  | ||||||
|  |   #if ENABLED(SDSORT_DYNAMIC_RAM) | ||||||
|  |     uint8_t *CardReader::sort_order; | ||||||
|  |   #else | ||||||
|  |     uint8_t CardReader::sort_order[SDSORT_LIMIT]; | ||||||
|  |   #endif | ||||||
|  |  | ||||||
|  |   #if ENABLED(SDSORT_USES_RAM) | ||||||
|  |  | ||||||
|  |     #if ENABLED(SDSORT_CACHE_NAMES) | ||||||
|  |       #if ENABLED(SDSORT_DYNAMIC_RAM) | ||||||
|  |         char **CardReader::sortshort, **CardReader::sortnames; | ||||||
|  |       #else | ||||||
|  |         char CardReader::sortshort[SDSORT_LIMIT][FILENAME_LENGTH + 1]; | ||||||
|  |         char CardReader::sortnames[SDSORT_LIMIT][SORTED_LONGNAME_MAXLEN + 1]; | ||||||
|  |       #endif | ||||||
|  |     #elif DISABLED(SDSORT_USES_STACK) | ||||||
|  |       char CardReader::sortnames[SDSORT_LIMIT][SORTED_LONGNAME_MAXLEN + 1]; | ||||||
|  |     #endif | ||||||
|  |  | ||||||
|  |     #if HAS_FOLDER_SORTING | ||||||
|  |       #if ENABLED(SDSORT_DYNAMIC_RAM) | ||||||
|  |         uint8_t *CardReader::isDir; | ||||||
|  |       #elif ENABLED(SDSORT_CACHE_NAMES) || DISABLED(SDSORT_USES_STACK) | ||||||
|  |         uint8_t CardReader::isDir[(SDSORT_LIMIT+7)>>3]; | ||||||
|  |       #endif | ||||||
|  |     #endif | ||||||
|  |  | ||||||
|  |   #endif // SDSORT_USES_RAM | ||||||
|  |  | ||||||
|  | #endif // SDCARD_SORT_ALPHA | ||||||
|  |  | ||||||
|  | Sd2Card CardReader::sd2card; | ||||||
|  | SdVolume CardReader::volume; | ||||||
|  | SdFile CardReader::file; | ||||||
|  |  | ||||||
|  | uint8_t CardReader::file_subcall_ctr; | ||||||
|  | uint32_t CardReader::filespos[SD_PROCEDURE_DEPTH]; | ||||||
|  | char CardReader::proc_filenames[SD_PROCEDURE_DEPTH][MAXPATHNAMELENGTH + 1]; | ||||||
|  |  | ||||||
|  | uint32_t CardReader::filesize, CardReader::sdpos; | ||||||
|  |  | ||||||
|  | LsAction CardReader::lsAction; //stored for recursion. | ||||||
|  | uint16_t CardReader::nrFiles; //counter for the files in the current directory and recycled as position counter for getting the nrFiles'th name in the directory. | ||||||
|  | char *CardReader::diveDirName; | ||||||
|  |  | ||||||
|  | #if ENABLED(AUTO_REPORT_SD_STATUS) | ||||||
|  |   uint8_t CardReader::auto_report_sd_interval; | ||||||
|  |   millis_t CardReader::next_sd_report_ms; | ||||||
|  |   #if NUM_SERIAL > 1 | ||||||
|  |     int8_t CardReader::serialport; | ||||||
|  |   #endif | ||||||
|  | #endif | ||||||
|  |  | ||||||
| CardReader::CardReader() { | CardReader::CardReader() { | ||||||
|   #if ENABLED(SDCARD_SORT_ALPHA) |   #if ENABLED(SDCARD_SORT_ALPHA) | ||||||
| @@ -56,9 +132,8 @@ CardReader::CardReader() { | |||||||
|       //sort_reverse = false; |       //sort_reverse = false; | ||||||
|     #endif |     #endif | ||||||
|   #endif |   #endif | ||||||
|   sdprinting = cardOK = saving = logging = false; |   flag.sdprinting = flag.cardOK = flag.saving = flag.logging = false; | ||||||
|   filesize = 0; |   filesize = sdpos = 0; | ||||||
|   sdpos = 0; |  | ||||||
|   file_subcall_ctr = 0; |   file_subcall_ctr = 0; | ||||||
|  |  | ||||||
|   workDirDepth = 0; |   workDirDepth = 0; | ||||||
| @@ -73,7 +148,7 @@ CardReader::CardReader() { | |||||||
|   #endif |   #endif | ||||||
| } | } | ||||||
|  |  | ||||||
| char *createFilename(char *buffer, const dir_t &p) { //buffer > 12characters | char *createFilename(char *buffer, const dir_t &p) { | ||||||
|   char *pos = buffer; |   char *pos = buffer; | ||||||
|   for (uint8_t i = 0; i < 11; i++) { |   for (uint8_t i = 0; i < 11; i++) { | ||||||
|     if (p.name[i] == ' ') continue; |     if (p.name[i] == ' ') continue; | ||||||
| @@ -108,7 +183,7 @@ void CardReader::lsDive(const char *prepend, SdFile parent, const char * const m | |||||||
|     if (DIR_IS_SUBDIR(&p) && lsAction != LS_Count && lsAction != LS_GetFilename) { |     if (DIR_IS_SUBDIR(&p) && lsAction != LS_Count && lsAction != LS_GetFilename) { | ||||||
|  |  | ||||||
|       // Get the short name for the item, which we know is a folder |       // Get the short name for the item, which we know is a folder | ||||||
|       char dosFilename[FILENAME_LENGTH]; |       char dosFilename[FILENAME_LENGTH + 1]; | ||||||
|       createFilename(dosFilename, p); |       createFilename(dosFilename, p); | ||||||
|  |  | ||||||
|       // Allocate enough stack space for the full path to a folder, trailing slash, and nul |       // Allocate enough stack space for the full path to a folder, trailing slash, and nul | ||||||
| @@ -120,7 +195,7 @@ void CardReader::lsDive(const char *prepend, SdFile parent, const char * const m | |||||||
|       // It contains the full path to the "parent" argument. |       // It contains the full path to the "parent" argument. | ||||||
|       // We now have the full path to the item in this folder. |       // We now have the full path to the item in this folder. | ||||||
|       strcpy(path, prepend_is_empty ? "/" : prepend); // root slash if prepend is empty |       strcpy(path, prepend_is_empty ? "/" : prepend); // root slash if prepend is empty | ||||||
|       strcat(path, dosFilename); // FILENAME_LENGTH-1 characters maximum |       strcat(path, dosFilename);                      // FILENAME_LENGTH characters maximum | ||||||
|       strcat(path, "/");                              // 1 character |       strcat(path, "/");                              // 1 character | ||||||
|  |  | ||||||
|       // Serial.print(path); |       // Serial.print(path); | ||||||
| @@ -150,9 +225,9 @@ void CardReader::lsDive(const char *prepend, SdFile parent, const char * const m | |||||||
|  |  | ||||||
|       if (!DIR_IS_FILE_OR_SUBDIR(&p) || (p.attributes & DIR_ATT_HIDDEN)) continue; |       if (!DIR_IS_FILE_OR_SUBDIR(&p) || (p.attributes & DIR_ATT_HIDDEN)) continue; | ||||||
|  |  | ||||||
|       filenameIsDir = DIR_IS_SUBDIR(&p); |       flag.filenameIsDir = DIR_IS_SUBDIR(&p); | ||||||
|  |  | ||||||
|       if (!filenameIsDir && (p.name[8] != 'G' || p.name[9] == '~')) continue; |       if (!flag.filenameIsDir && (p.name[8] != 'G' || p.name[9] == '~')) continue; | ||||||
|  |  | ||||||
|       switch (lsAction) {  // 1 based file count |       switch (lsAction) {  // 1 based file count | ||||||
|         case LS_Count: |         case LS_Count: | ||||||
| @@ -242,7 +317,7 @@ void CardReader::ls( | |||||||
|       SERIAL_PROTOCOL_P(port, longFilename[0] ? longFilename : "???"); |       SERIAL_PROTOCOL_P(port, longFilename[0] ? longFilename : "???"); | ||||||
|  |  | ||||||
|       // If the filename was printed then that's it |       // If the filename was printed then that's it | ||||||
|       if (!filenameIsDir) break; |       if (!flag.filenameIsDir) break; | ||||||
|  |  | ||||||
|       // SERIAL_ECHOPGM_P(port, "Opening dir: "); SERIAL_ECHOLN_P(port, segment); |       // SERIAL_ECHOPGM_P(port, "Opening dir: "); SERIAL_ECHOLN_P(port, segment); | ||||||
|  |  | ||||||
| @@ -275,7 +350,7 @@ void CardReader::printFilename( | |||||||
|   #endif |   #endif | ||||||
| ) { | ) { | ||||||
|   if (file.isOpen()) { |   if (file.isOpen()) { | ||||||
|     char dosFilename[FILENAME_LENGTH]; |     char dosFilename[FILENAME_LENGTH + 1]; | ||||||
|     file.getFilename(dosFilename); |     file.getFilename(dosFilename); | ||||||
|     SERIAL_ECHO_P(port, dosFilename); |     SERIAL_ECHO_P(port, dosFilename); | ||||||
|     #if ENABLED(LONG_FILENAME_HOST_SUPPORT) |     #if ENABLED(LONG_FILENAME_HOST_SUPPORT) | ||||||
| @@ -293,7 +368,7 @@ void CardReader::printFilename( | |||||||
| } | } | ||||||
|  |  | ||||||
| void CardReader::initsd() { | void CardReader::initsd() { | ||||||
|   cardOK = false; |   flag.cardOK = false; | ||||||
|   if (root.isOpen()) root.close(); |   if (root.isOpen()) root.close(); | ||||||
|  |  | ||||||
|   #ifndef SPI_SPEED |   #ifndef SPI_SPEED | ||||||
| @@ -318,7 +393,7 @@ void CardReader::initsd() { | |||||||
|     SERIAL_ERRORLNPGM(MSG_SD_OPENROOT_FAIL); |     SERIAL_ERRORLNPGM(MSG_SD_OPENROOT_FAIL); | ||||||
|   } |   } | ||||||
|   else { |   else { | ||||||
|     cardOK = true; |     flag.cardOK = true; | ||||||
|     SERIAL_ECHO_START(); |     SERIAL_ECHO_START(); | ||||||
|     SERIAL_ECHOLNPGM(MSG_SD_CARD_OK); |     SERIAL_ECHOLNPGM(MSG_SD_CARD_OK); | ||||||
|   } |   } | ||||||
| @@ -327,7 +402,7 @@ void CardReader::initsd() { | |||||||
|  |  | ||||||
| void CardReader::release() { | void CardReader::release() { | ||||||
|   stopSDPrint(); |   stopSDPrint(); | ||||||
|   cardOK = false; |   flag.cardOK = false; | ||||||
| } | } | ||||||
|  |  | ||||||
| void CardReader::openAndPrintFile(const char *name) { | void CardReader::openAndPrintFile(const char *name) { | ||||||
| @@ -339,8 +414,8 @@ void CardReader::openAndPrintFile(const char *name) { | |||||||
| } | } | ||||||
|  |  | ||||||
| void CardReader::startFileprint() { | void CardReader::startFileprint() { | ||||||
|   if (cardOK) { |   if (flag.cardOK) { | ||||||
|     sdprinting = true; |     flag.sdprinting = true; | ||||||
|     #if SD_RESORT |     #if SD_RESORT | ||||||
|       flush_presort(); |       flush_presort(); | ||||||
|     #endif |     #endif | ||||||
| @@ -355,7 +430,7 @@ void CardReader::stopSDPrint( | |||||||
|   #if ENABLED(ADVANCED_PAUSE_FEATURE) |   #if ENABLED(ADVANCED_PAUSE_FEATURE) | ||||||
|     did_pause_print = 0; |     did_pause_print = 0; | ||||||
|   #endif |   #endif | ||||||
|   sdprinting = abort_sd_printing = false; |   flag.sdprinting = flag.abort_sd_printing = false; | ||||||
|   if (isFileOpen()) file.close(); |   if (isFileOpen()) file.close(); | ||||||
|   #if SD_RESORT |   #if SD_RESORT | ||||||
|     if (re_sort) presort(); |     if (re_sort) presort(); | ||||||
| @@ -363,7 +438,7 @@ void CardReader::stopSDPrint( | |||||||
| } | } | ||||||
|  |  | ||||||
| void CardReader::openLogFile(char * const path) { | void CardReader::openLogFile(char * const path) { | ||||||
|   logging = true; |   flag.logging = true; | ||||||
|   openFile(path, false); |   openFile(path, false); | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -380,7 +455,7 @@ void CardReader::getAbsFilename(char *t) { | |||||||
|   for (uint8_t i = 0; i < workDirDepth; i++)                // Loop to current work dir |   for (uint8_t i = 0; i < workDirDepth; i++)                // Loop to current work dir | ||||||
|     appendAtom(workDirParents[i], t, cnt); |     appendAtom(workDirParents[i], t, cnt); | ||||||
|  |  | ||||||
|   if (cnt < MAXPATHNAMELENGTH - (FILENAME_LENGTH)) { |   if (cnt < MAXPATHNAMELENGTH - (FILENAME_LENGTH) - 1) {    // Leave room for filename and nul | ||||||
|     appendAtom(file, t, cnt); |     appendAtom(file, t, cnt); | ||||||
|     --t; |     --t; | ||||||
|   } |   } | ||||||
| @@ -389,7 +464,7 @@ void CardReader::getAbsFilename(char *t) { | |||||||
|  |  | ||||||
| void CardReader::openFile(char * const path, const bool read, const bool subcall/*=false*/) { | void CardReader::openFile(char * const path, const bool read, const bool subcall/*=false*/) { | ||||||
|  |  | ||||||
|   if (!cardOK) return; |   if (!flag.cardOK) return; | ||||||
|  |  | ||||||
|   uint8_t doing = 0; |   uint8_t doing = 0; | ||||||
|   if (isFileOpen()) {                     // Replacing current file or doing a subroutine |   if (isFileOpen()) {                     // Replacing current file or doing a subroutine | ||||||
| @@ -464,7 +539,7 @@ void CardReader::openFile(char * const path, const bool read, const bool subcall | |||||||
|       SERIAL_EOL(); |       SERIAL_EOL(); | ||||||
|     } |     } | ||||||
|     else { |     else { | ||||||
|       saving = true; |       flag.saving = true; | ||||||
|       getfilename(0, fname); |       getfilename(0, fname); | ||||||
|       #if ENABLED(EMERGENCY_PARSER) |       #if ENABLED(EMERGENCY_PARSER) | ||||||
|         emergency_parser.disable(); |         emergency_parser.disable(); | ||||||
| @@ -476,7 +551,7 @@ void CardReader::openFile(char * const path, const bool read, const bool subcall | |||||||
| } | } | ||||||
|  |  | ||||||
| void CardReader::removeFile(const char * const name) { | void CardReader::removeFile(const char * const name) { | ||||||
|   if (!cardOK) return; |   if (!flag.cardOK) return; | ||||||
|  |  | ||||||
|   //stopSDPrint(); |   //stopSDPrint(); | ||||||
|  |  | ||||||
| @@ -504,7 +579,7 @@ void CardReader::getStatus( | |||||||
|     const int8_t port/*= -1*/ |     const int8_t port/*= -1*/ | ||||||
|   #endif |   #endif | ||||||
| ) { | ) { | ||||||
|   if (cardOK && sdprinting) { |   if (flag.cardOK && flag.sdprinting) { | ||||||
|     SERIAL_PROTOCOLPGM_P(port, MSG_SD_PRINTING_BYTE); |     SERIAL_PROTOCOLPGM_P(port, MSG_SD_PRINTING_BYTE); | ||||||
|     SERIAL_PROTOCOL_P(port, sdpos); |     SERIAL_PROTOCOL_P(port, sdpos); | ||||||
|     SERIAL_PROTOCOLCHAR_P(port, '/'); |     SERIAL_PROTOCOLCHAR_P(port, '/'); | ||||||
| @@ -543,11 +618,11 @@ void CardReader::write_command(char *buf) { | |||||||
|  |  | ||||||
| void CardReader::checkautostart() { | void CardReader::checkautostart() { | ||||||
|  |  | ||||||
|   if (autostart_index < 0 || sdprinting) return; |   if (autostart_index < 0 || flag.sdprinting) return; | ||||||
|  |  | ||||||
|   if (!cardOK) initsd(); |   if (!flag.cardOK) initsd(); | ||||||
|  |  | ||||||
|   if (cardOK |   if (flag.cardOK | ||||||
|     #if ENABLED(POWER_LOSS_RECOVERY) |     #if ENABLED(POWER_LOSS_RECOVERY) | ||||||
|       && !recovery.valid() // Don't run auto#.g when a resume file exists |       && !recovery.valid() // Don't run auto#.g when a resume file exists | ||||||
|     #endif |     #endif | ||||||
| @@ -576,7 +651,7 @@ void CardReader::beginautostart() { | |||||||
| void CardReader::closefile(const bool store_location) { | void CardReader::closefile(const bool store_location) { | ||||||
|   file.sync(); |   file.sync(); | ||||||
|   file.close(); |   file.close(); | ||||||
|   saving = logging = false; |   flag.saving = flag.logging = false; | ||||||
|   sdpos = 0; |   sdpos = 0; | ||||||
|   #if ENABLED(EMERGENCY_PARSER) |   #if ENABLED(EMERGENCY_PARSER) | ||||||
|     emergency_parser.enable(); |     emergency_parser.enable(); | ||||||
| @@ -603,7 +678,7 @@ void CardReader::getfilename(uint16_t nr, const char * const match/*=NULL*/) { | |||||||
|     if (nr < sort_count) { |     if (nr < sort_count) { | ||||||
|       strcpy(filename, sortshort[nr]); |       strcpy(filename, sortshort[nr]); | ||||||
|       strcpy(longFilename, sortnames[nr]); |       strcpy(longFilename, sortnames[nr]); | ||||||
|       filenameIsDir = TEST(isDir[nr>>3], nr & 0x07); |       flag.filenameIsDir = TEST(isDir[nr>>3], nr & 0x07); | ||||||
|       return; |       return; | ||||||
|     } |     } | ||||||
|   #endif // SDSORT_CACHE_NAMES |   #endif // SDSORT_CACHE_NAMES | ||||||
| @@ -712,6 +787,33 @@ void CardReader::setroot() { | |||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   #if ENABLED(SDSORT_USES_RAM) | ||||||
|  |     #if ENABLED(SDSORT_DYNAMIC_RAM) | ||||||
|  |       // Use dynamic method to copy long filename | ||||||
|  |       #define SET_SORTNAME(I) (sortnames[I] = strdup(longest_filename())) | ||||||
|  |       #if ENABLED(SDSORT_CACHE_NAMES) | ||||||
|  |         // When caching also store the short name, since | ||||||
|  |         // we're replacing the getfilename() behavior. | ||||||
|  |         #define SET_SORTSHORT(I) (sortshort[I] = strdup(filename)) | ||||||
|  |       #else | ||||||
|  |         #define SET_SORTSHORT(I) NOOP | ||||||
|  |       #endif | ||||||
|  |     #else | ||||||
|  |       // Copy filenames into the static array | ||||||
|  |       #if SORTED_LONGNAME_MAXLEN != LONG_FILENAME_LENGTH | ||||||
|  |         #define SET_SORTNAME(I) do{ strncpy(sortnames[I], longest_filename(), SORTED_LONGNAME_MAXLEN); \ | ||||||
|  |                                     sortnames[I][SORTED_LONGNAME_MAXLEN] = '\0'; }while(0) | ||||||
|  |       #else | ||||||
|  |         #define SET_SORTNAME(I) strncpy(sortnames[I], longest_filename(), SORTED_LONGNAME_MAXLEN) | ||||||
|  |       #endif | ||||||
|  |       #if ENABLED(SDSORT_CACHE_NAMES) | ||||||
|  |         #define SET_SORTSHORT(I) strcpy(sortshort[I], filename) | ||||||
|  |       #else | ||||||
|  |         #define SET_SORTSHORT(I) NOOP | ||||||
|  |       #endif | ||||||
|  |     #endif | ||||||
|  |   #endif | ||||||
|  |  | ||||||
|   /** |   /** | ||||||
|    * Read all the files and produce a sort key |    * Read all the files and produce a sort key | ||||||
|    * |    * | ||||||
| @@ -754,7 +856,7 @@ void CardReader::setroot() { | |||||||
|             sortnames = new char*[fileCnt]; |             sortnames = new char*[fileCnt]; | ||||||
|           #endif |           #endif | ||||||
|         #elif ENABLED(SDSORT_USES_STACK) |         #elif ENABLED(SDSORT_USES_STACK) | ||||||
|           char sortnames[fileCnt][SORTED_LONGNAME_MAXLEN]; |           char sortnames[fileCnt][SORTED_LONGNAME_MAXLEN + 1]; | ||||||
|         #endif |         #endif | ||||||
|  |  | ||||||
|         // Folder sorting needs 1 bit per entry for flags. |         // Folder sorting needs 1 bit per entry for flags. | ||||||
| @@ -783,33 +885,15 @@ void CardReader::setroot() { | |||||||
|           // If using RAM then read all filenames now. |           // If using RAM then read all filenames now. | ||||||
|           #if ENABLED(SDSORT_USES_RAM) |           #if ENABLED(SDSORT_USES_RAM) | ||||||
|             getfilename(i); |             getfilename(i); | ||||||
|             #if ENABLED(SDSORT_DYNAMIC_RAM) |             SET_SORTNAME(i); | ||||||
|               // Use dynamic method to copy long filename |             SET_SORTSHORT(i); | ||||||
|               sortnames[i] = strdup(longest_filename()); |  | ||||||
|               #if ENABLED(SDSORT_CACHE_NAMES) |  | ||||||
|                 // When caching also store the short name, since |  | ||||||
|                 // we're replacing the getfilename() behavior. |  | ||||||
|                 sortshort[i] = strdup(filename); |  | ||||||
|               #endif |  | ||||||
|             #else |  | ||||||
|               // Copy filenames into the static array |  | ||||||
|               #if SORTED_LONGNAME_MAXLEN != LONG_FILENAME_LENGTH |  | ||||||
|                 strncpy(sortnames[i], longest_filename(), SORTED_LONGNAME_MAXLEN); |  | ||||||
|                 sortnames[i][SORTED_LONGNAME_MAXLEN - 1] = '\0'; |  | ||||||
|               #else |  | ||||||
|                 strncpy(sortnames[i], longest_filename(), SORTED_LONGNAME_MAXLEN); |  | ||||||
|               #endif |  | ||||||
|               #if ENABLED(SDSORT_CACHE_NAMES) |  | ||||||
|                 strcpy(sortshort[i], filename); |  | ||||||
|               #endif |  | ||||||
|             #endif |  | ||||||
|             // char out[30]; |             // char out[30]; | ||||||
|             // sprintf_P(out, PSTR("---- %i %s %s"), i, filenameIsDir ? "D" : " ", sortnames[i]); |             // sprintf_P(out, PSTR("---- %i %s %s"), i, flag.filenameIsDir ? "D" : " ", sortnames[i]); | ||||||
|             // SERIAL_ECHOLN(out); |             // SERIAL_ECHOLN(out); | ||||||
|             #if HAS_FOLDER_SORTING |             #if HAS_FOLDER_SORTING | ||||||
|               const uint16_t bit = i & 0x07, ind = i >> 3; |               const uint16_t bit = i & 0x07, ind = i >> 3; | ||||||
|               if (bit == 0) isDir[ind] = 0x00; |               if (bit == 0) isDir[ind] = 0x00; | ||||||
|               if (filenameIsDir) isDir[ind] |= _BV(bit); |               if (flag.filenameIsDir) isDir[ind] |= _BV(bit); | ||||||
|             #endif |             #endif | ||||||
|           #endif |           #endif | ||||||
|         } |         } | ||||||
| @@ -837,7 +921,7 @@ void CardReader::setroot() { | |||||||
|                     ? _SORT_CMP_NODIR() \ |                     ? _SORT_CMP_NODIR() \ | ||||||
|                     : (isDir[fs > 0 ? ind1 : ind2] & (fs > 0 ? _BV(bit1) : _BV(bit2))) != 0) |                     : (isDir[fs > 0 ? ind1 : ind2] & (fs > 0 ? _BV(bit1) : _BV(bit2))) != 0) | ||||||
|               #else |               #else | ||||||
|                 #define _SORT_CMP_DIR(fs) ((dir1 == filenameIsDir) ? _SORT_CMP_NODIR() : (fs > 0 ? dir1 : !dir1)) |                 #define _SORT_CMP_DIR(fs) ((dir1 == flag.filenameIsDir) ? _SORT_CMP_NODIR() : (fs > 0 ? dir1 : !dir1)) | ||||||
|               #endif |               #endif | ||||||
|             #endif |             #endif | ||||||
|  |  | ||||||
| @@ -847,7 +931,7 @@ void CardReader::setroot() { | |||||||
|               getfilename(o1); |               getfilename(o1); | ||||||
|               strcpy(name1, longest_filename()); // save (or getfilename below will trounce it) |               strcpy(name1, longest_filename()); // save (or getfilename below will trounce it) | ||||||
|               #if HAS_FOLDER_SORTING |               #if HAS_FOLDER_SORTING | ||||||
|                 bool dir1 = filenameIsDir; |                 bool dir1 = flag.filenameIsDir; | ||||||
|               #endif |               #endif | ||||||
|               getfilename(o2); |               getfilename(o2); | ||||||
|               char *name2 = longest_filename(); // use the string in-place |               char *name2 = longest_filename(); // use the string in-place | ||||||
| @@ -885,27 +969,17 @@ void CardReader::setroot() { | |||||||
|       else { |       else { | ||||||
|         sort_order[0] = 0; |         sort_order[0] = 0; | ||||||
|         #if ENABLED(SDSORT_USES_RAM) && ENABLED(SDSORT_CACHE_NAMES) |         #if ENABLED(SDSORT_USES_RAM) && ENABLED(SDSORT_CACHE_NAMES) | ||||||
|           getfilename(0); |  | ||||||
|           #if ENABLED(SDSORT_DYNAMIC_RAM) |           #if ENABLED(SDSORT_DYNAMIC_RAM) | ||||||
|             sortnames = new char*[1]; |             sortnames = new char*[1]; | ||||||
|             sortnames[0] = strdup(longest_filename()); // malloc |  | ||||||
|             #if ENABLED(SDSORT_CACHE_NAMES) |             #if ENABLED(SDSORT_CACHE_NAMES) | ||||||
|               sortshort = new char*[1]; |               sortshort = new char*[1]; | ||||||
|               sortshort[0] = strdup(filename);       // malloc |  | ||||||
|             #endif |             #endif | ||||||
|             isDir = new uint8_t[1]; |             isDir = new uint8_t[1]; | ||||||
|           #else |  | ||||||
|             #if SORTED_LONGNAME_MAXLEN != LONG_FILENAME_LENGTH |  | ||||||
|               strncpy(sortnames[0], longest_filename(), SORTED_LONGNAME_MAXLEN); |  | ||||||
|               sortnames[0][SORTED_LONGNAME_MAXLEN - 1] = '\0'; |  | ||||||
|             #else |  | ||||||
|               strncpy(sortnames[0], longest_filename(), SORTED_LONGNAME_MAXLEN); |  | ||||||
|           #endif |           #endif | ||||||
|             #if ENABLED(SDSORT_CACHE_NAMES) |           getfilename(0); | ||||||
|               strcpy(sortshort[0], filename); |           SET_SORTNAME(0); | ||||||
|             #endif |           SET_SORTSHORT(0); | ||||||
|           #endif |           isDir[0] = flag.filenameIsDir ? 0x01 : 0x00; | ||||||
|           isDir[0] = filenameIsDir ? 0x01 : 0x00; |  | ||||||
|         #endif |         #endif | ||||||
|       } |       } | ||||||
|  |  | ||||||
| @@ -1007,7 +1081,7 @@ void CardReader::printingHasFinished() { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   void CardReader::openJobRecoveryFile(const bool read) { |   void CardReader::openJobRecoveryFile(const bool read) { | ||||||
|     if (!cardOK) return; |     if (!flag.cardOK) return; | ||||||
|     if (recovery.file.isOpen()) return; |     if (recovery.file.isOpen()) return; | ||||||
|     if (!recovery.file.open(&root, job_recovery_file_name, read ? O_READ : O_CREAT | O_WRITE | O_TRUNC | O_SYNC)) { |     if (!recovery.file.open(&root, job_recovery_file_name, read ? O_READ : O_CREAT | O_WRITE | O_TRUNC | O_SYNC)) { | ||||||
|       SERIAL_PROTOCOLPAIR(MSG_SD_OPEN_FILE_FAIL, job_recovery_file_name); |       SERIAL_PROTOCOLPAIR(MSG_SD_OPEN_FILE_FAIL, job_recovery_file_name); | ||||||
|   | |||||||
| @@ -28,105 +28,120 @@ | |||||||
| #define SD_RESORT ENABLED(SDCARD_SORT_ALPHA) && ENABLED(SDSORT_DYNAMIC_RAM) | #define SD_RESORT ENABLED(SDCARD_SORT_ALPHA) && ENABLED(SDSORT_DYNAMIC_RAM) | ||||||
|  |  | ||||||
| #define MAX_DIR_DEPTH     10       // Maximum folder depth | #define MAX_DIR_DEPTH     10       // Maximum folder depth | ||||||
|  | #define MAXDIRNAMELENGTH   8       // DOS folder name size | ||||||
|  | #define MAXPATHNAMELENGTH  (1 + (MAXDIRNAMELENGTH + 1) * (MAX_DIR_DEPTH) + 1 + FILENAME_LENGTH) // "/" + N * ("ADIRNAME/") + "filename.ext" | ||||||
|  |  | ||||||
| #include "SdFile.h" | #include "SdFile.h" | ||||||
|  |  | ||||||
| enum LsAction : uint8_t { LS_SerialPrint, LS_Count, LS_GetFilename }; | enum LsAction : uint8_t { LS_SerialPrint, LS_Count, LS_GetFilename }; | ||||||
|  |  | ||||||
|  | typedef struct { | ||||||
|  |   bool saving:1, | ||||||
|  |        logging:1, | ||||||
|  |        sdprinting:1, | ||||||
|  |        cardOK:1, | ||||||
|  |        filenameIsDir:1, | ||||||
|  |        abort_sd_printing:1 | ||||||
|  |        #if ENABLED(FAST_FILE_TRANSFER) | ||||||
|  |          , binary_mode:1 | ||||||
|  |        #endif | ||||||
|  |     ; | ||||||
|  | } card_flags_t; | ||||||
|  |  | ||||||
| class CardReader { | class CardReader { | ||||||
| public: | public: | ||||||
|   CardReader(); |   CardReader(); | ||||||
|  |  | ||||||
|   void initsd(); |   static void initsd(); | ||||||
|   void write_command(char *buf); |   static void write_command(char *buf); | ||||||
|  |  | ||||||
|   void beginautostart(); |   static void beginautostart(); | ||||||
|   void checkautostart(); |   static void checkautostart(); | ||||||
|  |  | ||||||
|   void openFile(char * const path, const bool read, const bool subcall=false); |   static void openFile(char * const path, const bool read, const bool subcall=false); | ||||||
|   void openLogFile(char * const path); |   static void openLogFile(char * const path); | ||||||
|   void removeFile(const char * const name); |   static void removeFile(const char * const name); | ||||||
|   void closefile(const bool store_location=false); |   static void closefile(const bool store_location=false); | ||||||
|   void release(); |   static void release(); | ||||||
|   void openAndPrintFile(const char *name); |   static void openAndPrintFile(const char *name); | ||||||
|   void startFileprint(); |   static void startFileprint(); | ||||||
|   void stopSDPrint( |   static void stopSDPrint( | ||||||
|     #if SD_RESORT |     #if SD_RESORT | ||||||
|       const bool re_sort=false |       const bool re_sort=false | ||||||
|     #endif |     #endif | ||||||
|   ); |   ); | ||||||
|   void getStatus( |   static void getStatus( | ||||||
|     #if NUM_SERIAL > 1 |     #if NUM_SERIAL > 1 | ||||||
|       const int8_t port = -1 |       const int8_t port = -1 | ||||||
|     #endif |     #endif | ||||||
|   ); |   ); | ||||||
|   void printingHasFinished(); |   static void printingHasFinished(); | ||||||
|   void printFilename( |   static void printFilename( | ||||||
|     #if NUM_SERIAL > 1 |     #if NUM_SERIAL > 1 | ||||||
|       const int8_t port = -1 |       const int8_t port = -1 | ||||||
|     #endif |     #endif | ||||||
|   ); |   ); | ||||||
|  |  | ||||||
|   #if ENABLED(LONG_FILENAME_HOST_SUPPORT) |   #if ENABLED(LONG_FILENAME_HOST_SUPPORT) | ||||||
|     void printLongPath(char *path |     static void printLongPath(char *path | ||||||
|       #if NUM_SERIAL > 1 |       #if NUM_SERIAL > 1 | ||||||
|         , const int8_t port = -1 |         , const int8_t port = -1 | ||||||
|       #endif |       #endif | ||||||
|     ); |     ); | ||||||
|   #endif |   #endif | ||||||
|  |  | ||||||
|   void getfilename(uint16_t nr, const char* const match=NULL); |   static void getfilename(uint16_t nr, const char* const match=NULL); | ||||||
|   uint16_t getnrfilenames(); |   static uint16_t getnrfilenames(); | ||||||
|  |  | ||||||
|   void getAbsFilename(char *t); |   static void getAbsFilename(char *t); | ||||||
|  |  | ||||||
|   void ls( |   static void ls( | ||||||
|     #if NUM_SERIAL > 1 |     #if NUM_SERIAL > 1 | ||||||
|       const int8_t port = -1 |       const int8_t port = -1 | ||||||
|     #endif |     #endif | ||||||
|   ); |   ); | ||||||
|   void chdir(const char *relpath); |   static void chdir(const char *relpath); | ||||||
|   int8_t updir(); |   static int8_t updir(); | ||||||
|   void setroot(); |   static void setroot(); | ||||||
|  |  | ||||||
|   const char* diveToFile(SdFile*& curDir, const char * const path, const bool echo); |   static const char* diveToFile(SdFile*& curDir, const char * const path, const bool echo); | ||||||
|  |  | ||||||
|   uint16_t get_num_Files(); |   static uint16_t get_num_Files(); | ||||||
|  |  | ||||||
|   #if ENABLED(SDCARD_SORT_ALPHA) |   #if ENABLED(SDCARD_SORT_ALPHA) | ||||||
|     void presort(); |     static void presort(); | ||||||
|     void getfilename_sorted(const uint16_t nr); |     static void getfilename_sorted(const uint16_t nr); | ||||||
|     #if ENABLED(SDSORT_GCODE) |     #if ENABLED(SDSORT_GCODE) | ||||||
|       FORCE_INLINE void setSortOn(bool b) { sort_alpha = b; presort(); } |       FORCE_INLINE static void setSortOn(bool b) { sort_alpha = b; presort(); } | ||||||
|       FORCE_INLINE void setSortFolders(int i) { sort_folders = i; presort(); } |       FORCE_INLINE static void setSortFolders(int i) { sort_folders = i; presort(); } | ||||||
|       //FORCE_INLINE void setSortReverse(bool b) { sort_reverse = b; } |       //FORCE_INLINE static void setSortReverse(bool b) { sort_reverse = b; } | ||||||
|     #endif |     #endif | ||||||
|   #else |   #else | ||||||
|     FORCE_INLINE void getfilename_sorted(const uint16_t nr) { getfilename(nr); } |     FORCE_INLINE static void getfilename_sorted(const uint16_t nr) { getfilename(nr); } | ||||||
|   #endif |   #endif | ||||||
|  |  | ||||||
|   #if ENABLED(POWER_LOSS_RECOVERY) |   #if ENABLED(POWER_LOSS_RECOVERY) | ||||||
|     bool jobRecoverFileExists(); |     static bool jobRecoverFileExists(); | ||||||
|     void openJobRecoveryFile(const bool read); |     static void openJobRecoveryFile(const bool read); | ||||||
|     void removeJobRecoveryFile(); |     static void removeJobRecoveryFile(); | ||||||
|   #endif |   #endif | ||||||
|  |  | ||||||
|   inline void pauseSDPrint() { sdprinting = false; } |   static inline void pauseSDPrint() { flag.sdprinting = false; } | ||||||
|   inline bool isFileOpen() { return file.isOpen(); } |   static inline bool isFileOpen() { return file.isOpen(); } | ||||||
|   inline bool eof() { return sdpos >= filesize; } |   static inline bool eof() { return sdpos >= filesize; } | ||||||
|   inline int16_t get() { sdpos = file.curPosition(); return (int16_t)file.read(); } |   static inline int16_t get() { sdpos = file.curPosition(); return (int16_t)file.read(); } | ||||||
|   inline void setIndex(const uint32_t index) { sdpos = index; file.seekSet(index); } |   static inline void setIndex(const uint32_t index) { sdpos = index; file.seekSet(index); } | ||||||
|   inline uint32_t getIndex() { return sdpos; } |   static inline uint32_t getIndex() { return sdpos; } | ||||||
|   inline uint8_t percentDone() { return (isFileOpen() && filesize) ? sdpos / ((filesize + 99) / 100) : 0; } |   static inline uint8_t percentDone() { return (isFileOpen() && filesize) ? sdpos / ((filesize + 99) / 100) : 0; } | ||||||
|   inline char* getWorkDirName() { workDir.getFilename(filename); return filename; } |   static inline char* getWorkDirName() { workDir.getFilename(filename); return filename; } | ||||||
|   inline int16_t read(void* buf, uint16_t nbyte) { return file.isOpen() ? file.read(buf, nbyte) : -1; } |   static inline int16_t read(void* buf, uint16_t nbyte) { return file.isOpen() ? file.read(buf, nbyte) : -1; } | ||||||
|   inline int16_t write(void* buf, uint16_t nbyte) { return file.isOpen() ? file.write(buf, nbyte) : -1; } |   static inline int16_t write(void* buf, uint16_t nbyte) { return file.isOpen() ? file.write(buf, nbyte) : -1; } | ||||||
|  |  | ||||||
|   Sd2Card& getSd2Card() { return sd2card; } |   static Sd2Card& getSd2Card() { return sd2card; } | ||||||
|  |  | ||||||
|   #if ENABLED(AUTO_REPORT_SD_STATUS) |   #if ENABLED(AUTO_REPORT_SD_STATUS) | ||||||
|     void auto_report_sd_status(void); |     static void auto_report_sd_status(void); | ||||||
|     inline void set_auto_report_interval(uint8_t v |     static inline void set_auto_report_interval(uint8_t v | ||||||
|       #if NUM_SERIAL > 1 |       #if NUM_SERIAL > 1 | ||||||
|         , int8_t port |         , int8_t port | ||||||
|       #endif |       #endif | ||||||
| @@ -140,40 +155,39 @@ public: | |||||||
|     } |     } | ||||||
|   #endif |   #endif | ||||||
|  |  | ||||||
|   inline char* longest_filename() { return longFilename[0] ? longFilename : filename; } |   static inline char* longest_filename() { return longFilename[0] ? longFilename : filename; } | ||||||
|  |  | ||||||
| public: | public: | ||||||
|   bool saving, logging, sdprinting, cardOK, filenameIsDir, abort_sd_printing; |   static card_flags_t flag; | ||||||
|   char filename[FILENAME_LENGTH], longFilename[LONG_FILENAME_LENGTH]; |   static char filename[FILENAME_LENGTH + 1], longFilename[LONG_FILENAME_LENGTH + 1]; | ||||||
|   int8_t autostart_index; |   static int8_t autostart_index; | ||||||
|  |  | ||||||
|   #if ENABLED(FAST_FILE_TRANSFER) |   #if ENABLED(FAST_FILE_TRANSFER) | ||||||
|     bool binary_mode; |  | ||||||
|     #if NUM_SERIAL > 1 |     #if NUM_SERIAL > 1 | ||||||
|       uint8_t transfer_port; |       static uint8_t transfer_port; | ||||||
|     #else |     #else | ||||||
|       static constexpr uint8_t transfer_port = 0; |       static constexpr uint8_t transfer_port = 0; | ||||||
|     #endif |     #endif | ||||||
|   #endif |   #endif | ||||||
|  |  | ||||||
| private: | private: | ||||||
|   SdFile root, workDir, workDirParents[MAX_DIR_DEPTH]; |   static SdFile root, workDir, workDirParents[MAX_DIR_DEPTH]; | ||||||
|   uint8_t workDirDepth; |   static uint8_t workDirDepth; | ||||||
|  |  | ||||||
|   // Sort files and folders alphabetically. |   // Sort files and folders alphabetically. | ||||||
|   #if ENABLED(SDCARD_SORT_ALPHA) |   #if ENABLED(SDCARD_SORT_ALPHA) | ||||||
|     uint16_t sort_count;        // Count of sorted items in the current directory |     static uint16_t sort_count;   // Count of sorted items in the current directory | ||||||
|     #if ENABLED(SDSORT_GCODE) |     #if ENABLED(SDSORT_GCODE) | ||||||
|       bool sort_alpha;          // Flag to enable / disable the feature |       static bool sort_alpha;     // Flag to enable / disable the feature | ||||||
|       int sort_folders;         // Flag to enable / disable folder sorting |       static int sort_folders;    // Folder sorting before/none/after | ||||||
|       //bool sort_reverse;      // Flag to enable / disable reverse sorting |       //static bool sort_reverse; // Flag to enable / disable reverse sorting | ||||||
|     #endif |     #endif | ||||||
|  |  | ||||||
|     // By default the sort index is static |     // By default the sort index is static | ||||||
|     #if ENABLED(SDSORT_DYNAMIC_RAM) |     #if ENABLED(SDSORT_DYNAMIC_RAM) | ||||||
|       uint8_t *sort_order; |       static uint8_t *sort_order; | ||||||
|     #else |     #else | ||||||
|       uint8_t sort_order[SDSORT_LIMIT]; |       static uint8_t sort_order[SDSORT_LIMIT]; | ||||||
|     #endif |     #endif | ||||||
|  |  | ||||||
|     #if ENABLED(SDSORT_USES_RAM) && ENABLED(SDSORT_CACHE_NAMES) && DISABLED(SDSORT_DYNAMIC_RAM) |     #if ENABLED(SDSORT_USES_RAM) && ENABLED(SDSORT_CACHE_NAMES) && DISABLED(SDSORT_DYNAMIC_RAM) | ||||||
| @@ -188,21 +202,21 @@ private: | |||||||
|       // If using dynamic ram for names, allocate on the heap. |       // If using dynamic ram for names, allocate on the heap. | ||||||
|       #if ENABLED(SDSORT_CACHE_NAMES) |       #if ENABLED(SDSORT_CACHE_NAMES) | ||||||
|         #if ENABLED(SDSORT_DYNAMIC_RAM) |         #if ENABLED(SDSORT_DYNAMIC_RAM) | ||||||
|           char **sortshort, **sortnames; |           static char **sortshort, **sortnames; | ||||||
|         #else |         #else | ||||||
|           char sortshort[SDSORT_LIMIT][FILENAME_LENGTH]; |           static char sortshort[SDSORT_LIMIT][FILENAME_LENGTH + 1]; | ||||||
|           char sortnames[SDSORT_LIMIT][SORTED_LONGNAME_MAXLEN]; |           static char sortnames[SDSORT_LIMIT][SORTED_LONGNAME_MAXLEN + 1]; | ||||||
|         #endif |         #endif | ||||||
|       #elif DISABLED(SDSORT_USES_STACK) |       #elif DISABLED(SDSORT_USES_STACK) | ||||||
|         char sortnames[SDSORT_LIMIT][SORTED_LONGNAME_MAXLEN]; |         static char sortnames[SDSORT_LIMIT][SORTED_LONGNAME_MAXLEN + 1]; | ||||||
|       #endif |       #endif | ||||||
|  |  | ||||||
|       // Folder sorting uses an isDir array when caching items. |       // Folder sorting uses an isDir array when caching items. | ||||||
|       #if HAS_FOLDER_SORTING |       #if HAS_FOLDER_SORTING | ||||||
|         #if ENABLED(SDSORT_DYNAMIC_RAM) |         #if ENABLED(SDSORT_DYNAMIC_RAM) | ||||||
|           uint8_t *isDir; |           static uint8_t *isDir; | ||||||
|         #elif ENABLED(SDSORT_CACHE_NAMES) || DISABLED(SDSORT_USES_STACK) |         #elif ENABLED(SDSORT_CACHE_NAMES) || DISABLED(SDSORT_USES_STACK) | ||||||
|           uint8_t isDir[(SDSORT_LIMIT+7)>>3]; |           static uint8_t isDir[(SDSORT_LIMIT+7)>>3]; | ||||||
|         #endif |         #endif | ||||||
|       #endif |       #endif | ||||||
|  |  | ||||||
| @@ -210,28 +224,31 @@ private: | |||||||
|  |  | ||||||
|   #endif // SDCARD_SORT_ALPHA |   #endif // SDCARD_SORT_ALPHA | ||||||
|  |  | ||||||
|   Sd2Card sd2card; |   static Sd2Card sd2card; | ||||||
|   SdVolume volume; |   static SdVolume volume; | ||||||
|   SdFile file; |   static SdFile file; | ||||||
|  |  | ||||||
|  |   #ifndef SD_PROCEDURE_DEPTH | ||||||
|     #define SD_PROCEDURE_DEPTH 1 |     #define SD_PROCEDURE_DEPTH 1 | ||||||
|   #define MAXPATHNAMELENGTH (FILENAME_LENGTH*MAX_DIR_DEPTH + MAX_DIR_DEPTH + 1) |   #endif | ||||||
|   uint8_t file_subcall_ctr; |  | ||||||
|   uint32_t filespos[SD_PROCEDURE_DEPTH]; |  | ||||||
|   char proc_filenames[SD_PROCEDURE_DEPTH][MAXPATHNAMELENGTH]; |  | ||||||
|   uint32_t filesize, sdpos; |  | ||||||
|  |  | ||||||
|   LsAction lsAction; //stored for recursion. |   static uint8_t file_subcall_ctr; | ||||||
|   uint16_t nrFiles; //counter for the files in the current directory and recycled as position counter for getting the nrFiles'th name in the directory. |   static uint32_t filespos[SD_PROCEDURE_DEPTH]; | ||||||
|   char* diveDirName; |   static char proc_filenames[SD_PROCEDURE_DEPTH][MAXPATHNAMELENGTH + 1]; | ||||||
|   void lsDive(const char *prepend, SdFile parent, const char * const match=NULL |  | ||||||
|  |   static uint32_t filesize, sdpos; | ||||||
|  |  | ||||||
|  |   static LsAction lsAction; //stored for recursion. | ||||||
|  |   static uint16_t nrFiles; //counter for the files in the current directory and recycled as position counter for getting the nrFiles'th name in the directory. | ||||||
|  |   static char *diveDirName; | ||||||
|  |   static void lsDive(const char *prepend, SdFile parent, const char * const match=NULL | ||||||
|     #if NUM_SERIAL > 1 |     #if NUM_SERIAL > 1 | ||||||
|       , const int8_t port = -1 |       , const int8_t port = -1 | ||||||
|     #endif |     #endif | ||||||
|   ); |   ); | ||||||
|  |  | ||||||
|   #if ENABLED(SDCARD_SORT_ALPHA) |   #if ENABLED(SDCARD_SORT_ALPHA) | ||||||
|     void flush_presort(); |     static void flush_presort(); | ||||||
|   #endif |   #endif | ||||||
|  |  | ||||||
|   #if ENABLED(AUTO_REPORT_SD_STATUS) |   #if ENABLED(AUTO_REPORT_SD_STATUS) | ||||||
| @@ -256,7 +273,7 @@ private: | |||||||
|   #define IS_SD_INSERTED() true |   #define IS_SD_INSERTED() true | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
| #define IS_SD_PRINTING()  card.sdprinting | #define IS_SD_PRINTING()  card.flag.sdprinting | ||||||
| #define IS_SD_FILE_OPEN() card.isFileOpen() | #define IS_SD_FILE_OPEN() card.isFileOpen() | ||||||
|  |  | ||||||
| extern CardReader card; | extern CardReader card; | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user