2016-03-25 06:19:46 +00:00
|
|
|
/**
|
2016-03-24 18:01:20 +00:00
|
|
|
* Marlin 3D Printer Firmware
|
|
|
|
* Copyright (C) 2016 MarlinFirmware [https://github.com/MarlinFirmware/Marlin]
|
|
|
|
*
|
|
|
|
* Based on Sprinter and grbl.
|
|
|
|
* Copyright (C) 2011 Camiel Gubbels / Erik van der Zalm
|
|
|
|
*
|
|
|
|
* 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 <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2017-11-15 06:06:20 +00:00
|
|
|
#ifndef _CARDREADER_H_
|
|
|
|
#define _CARDREADER_H_
|
2011-11-06 20:39:53 +00:00
|
|
|
|
2018-03-07 07:42:11 +00:00
|
|
|
#include "../inc/MarlinConfig.h"
|
|
|
|
|
|
|
|
#if ENABLED(SDSUPPORT)
|
|
|
|
|
|
|
|
#define SD_RESORT ENABLED(SDCARD_SORT_ALPHA) && ENABLED(SDSORT_DYNAMIC_RAM)
|
|
|
|
|
2015-03-02 15:06:01 +00:00
|
|
|
#define MAX_DIR_DEPTH 10 // Maximum folder depth
|
2013-05-04 11:18:02 +00:00
|
|
|
|
2011-11-19 12:13:34 +00:00
|
|
|
#include "SdFile.h"
|
2016-08-03 02:36:58 +00:00
|
|
|
|
2015-03-02 15:06:01 +00:00
|
|
|
class CardReader {
|
2011-11-06 20:39:53 +00:00
|
|
|
public:
|
|
|
|
CardReader();
|
2015-03-02 15:06:01 +00:00
|
|
|
|
2011-11-06 20:39:53 +00:00
|
|
|
void initsd();
|
|
|
|
void write_command(char *buf);
|
2017-11-15 06:06:20 +00:00
|
|
|
// Files auto[0-9].g on the sd card are performed in sequence.
|
|
|
|
// This is to delay autostart and hence the initialisation of
|
|
|
|
// the sd card to some seconds after the normal init, so the
|
|
|
|
// device is available soon after a reset.
|
2011-11-06 20:39:53 +00:00
|
|
|
|
2015-03-02 15:06:01 +00:00
|
|
|
void checkautostart(bool x);
|
2017-11-15 06:15:57 +00:00
|
|
|
void openFile(char* name, const bool read, const bool subcall=false);
|
2013-03-31 23:59:16 +00:00
|
|
|
void openLogFile(char* name);
|
2017-11-15 06:06:20 +00:00
|
|
|
void removeFile(const char * const name);
|
2013-10-22 08:04:08 +00:00
|
|
|
void closefile(bool store_location=false);
|
2011-11-06 21:48:15 +00:00
|
|
|
void release();
|
2016-02-21 01:35:35 +00:00
|
|
|
void openAndPrintFile(const char *name);
|
2011-11-06 21:48:15 +00:00
|
|
|
void startFileprint();
|
2018-03-07 07:42:11 +00:00
|
|
|
void stopSDPrint(
|
|
|
|
#if SD_RESORT
|
|
|
|
const bool re_sort=false
|
|
|
|
#endif
|
|
|
|
);
|
2017-11-05 14:49:38 +00:00
|
|
|
void getStatus(
|
|
|
|
#if NUM_SERIAL > 1
|
|
|
|
const int8_t port = -1
|
|
|
|
#endif
|
|
|
|
);
|
2011-11-26 10:51:38 +00:00
|
|
|
void printingHasFinished();
|
2018-03-11 10:57:31 +00:00
|
|
|
void printFilename(
|
|
|
|
#if NUM_SERIAL > 1
|
|
|
|
const int8_t port = -1
|
|
|
|
#endif
|
|
|
|
);
|
2011-11-19 19:18:54 +00:00
|
|
|
|
2015-07-31 05:21:18 +00:00
|
|
|
#if ENABLED(LONG_FILENAME_HOST_SUPPORT)
|
2017-11-05 14:49:38 +00:00
|
|
|
void printLongPath(char *path
|
|
|
|
#if NUM_SERIAL > 1
|
|
|
|
, const int8_t port = -1
|
|
|
|
#endif
|
|
|
|
);
|
2015-05-18 00:36:32 +00:00
|
|
|
#endif
|
|
|
|
|
2014-12-18 07:07:36 +00:00
|
|
|
void getfilename(uint16_t nr, const char* const match=NULL);
|
2011-11-19 12:13:34 +00:00
|
|
|
uint16_t getnrfilenames();
|
2015-03-02 15:06:01 +00:00
|
|
|
|
2013-10-22 08:02:18 +00:00
|
|
|
void getAbsFilename(char *t);
|
2011-11-06 21:48:15 +00:00
|
|
|
|
2017-11-05 14:49:38 +00:00
|
|
|
void ls(
|
|
|
|
#if NUM_SERIAL > 1
|
|
|
|
const int8_t port = -1
|
|
|
|
#endif
|
|
|
|
);
|
2015-10-03 06:08:58 +00:00
|
|
|
void chdir(const char *relpath);
|
2017-11-26 07:14:18 +00:00
|
|
|
int8_t updir();
|
2011-12-26 08:20:33 +00:00
|
|
|
void setroot();
|
|
|
|
|
2017-07-30 05:02:41 +00:00
|
|
|
uint16_t get_num_Files();
|
2017-09-06 11:28:32 +00:00
|
|
|
|
2017-02-09 13:02:25 +00:00
|
|
|
#if ENABLED(SDCARD_SORT_ALPHA)
|
|
|
|
void presort();
|
|
|
|
void getfilename_sorted(const uint16_t nr);
|
|
|
|
#if ENABLED(SDSORT_GCODE)
|
|
|
|
FORCE_INLINE void setSortOn(bool b) { sort_alpha = b; presort(); }
|
|
|
|
FORCE_INLINE void setSortFolders(int i) { sort_folders = i; presort(); }
|
|
|
|
//FORCE_INLINE void setSortReverse(bool b) { sort_reverse = b; }
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
2018-04-22 00:41:26 +00:00
|
|
|
#if ENABLED(POWER_LOSS_RECOVERY)
|
|
|
|
void openJobRecoveryFile(const bool read);
|
|
|
|
void closeJobRecoveryFile();
|
|
|
|
bool jobRecoverFileExists();
|
|
|
|
int16_t saveJobRecoveryInfo();
|
|
|
|
int16_t loadJobRecoveryInfo();
|
|
|
|
void removeJobRecoveryFile();
|
|
|
|
#endif
|
|
|
|
|
2016-08-12 10:21:10 +00:00
|
|
|
FORCE_INLINE void pauseSDPrint() { sdprinting = false; }
|
2012-12-12 10:47:03 +00:00
|
|
|
FORCE_INLINE bool isFileOpen() { return file.isOpen(); }
|
2015-03-02 15:06:01 +00:00
|
|
|
FORCE_INLINE bool eof() { return sdpos >= filesize; }
|
|
|
|
FORCE_INLINE int16_t get() { sdpos = file.curPosition(); return (int16_t)file.read(); }
|
2018-04-22 00:41:26 +00:00
|
|
|
FORCE_INLINE void setIndex(const uint32_t index) { sdpos = index; file.seekSet(index); }
|
|
|
|
FORCE_INLINE uint32_t getIndex() { return sdpos; }
|
2015-03-02 15:06:01 +00:00
|
|
|
FORCE_INLINE uint8_t percentDone() { return (isFileOpen() && filesize) ? sdpos / ((filesize + 99) / 100) : 0; }
|
|
|
|
FORCE_INLINE char* getWorkDirName() { workDir.getFilename(filename); return filename; }
|
2011-11-06 20:39:53 +00:00
|
|
|
|
2017-12-21 05:30:18 +00:00
|
|
|
Sd2Card& getSd2Card() { return card; }
|
2018-02-26 21:38:27 +00:00
|
|
|
|
|
|
|
#if ENABLED(AUTO_REPORT_SD_STATUS)
|
|
|
|
void auto_report_sd_status(void);
|
|
|
|
FORCE_INLINE void set_auto_report_interval(uint8_t v
|
|
|
|
#if NUM_SERIAL > 1
|
|
|
|
, int8_t port
|
|
|
|
#endif
|
|
|
|
) {
|
|
|
|
#if NUM_SERIAL > 1
|
|
|
|
serialport = port;
|
|
|
|
#endif
|
|
|
|
NOMORE(v, 60);
|
|
|
|
auto_report_sd_interval = v;
|
|
|
|
next_sd_report_ms = millis() + 1000UL * v;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2011-11-06 20:39:53 +00:00
|
|
|
public:
|
2015-03-02 15:06:01 +00:00
|
|
|
bool saving, logging, sdprinting, cardOK, filenameIsDir;
|
|
|
|
char filename[FILENAME_LENGTH], longFilename[LONG_FILENAME_LENGTH];
|
2015-01-24 06:55:13 +00:00
|
|
|
int autostart_index;
|
2011-11-06 21:48:15 +00:00
|
|
|
private:
|
2015-03-02 15:06:01 +00:00
|
|
|
SdFile root, *curDir, workDir, workDirParents[MAX_DIR_DEPTH];
|
2016-03-29 10:18:53 +00:00
|
|
|
uint8_t workDirDepth;
|
2017-02-09 13:02:25 +00:00
|
|
|
|
|
|
|
// Sort files and folders alphabetically.
|
|
|
|
#if ENABLED(SDCARD_SORT_ALPHA)
|
|
|
|
uint16_t sort_count; // Count of sorted items in the current directory
|
|
|
|
#if ENABLED(SDSORT_GCODE)
|
|
|
|
bool sort_alpha; // Flag to enable / disable the feature
|
|
|
|
int sort_folders; // Flag to enable / disable folder sorting
|
|
|
|
//bool sort_reverse; // Flag to enable / disable reverse sorting
|
|
|
|
#endif
|
|
|
|
|
2017-02-09 13:05:34 +00:00
|
|
|
// By default the sort index is static
|
|
|
|
#if ENABLED(SDSORT_DYNAMIC_RAM)
|
|
|
|
uint8_t *sort_order;
|
|
|
|
#else
|
|
|
|
uint8_t sort_order[SDSORT_LIMIT];
|
|
|
|
#endif
|
2017-02-09 13:02:25 +00:00
|
|
|
|
2017-10-19 03:18:51 +00:00
|
|
|
#if ENABLED(SDSORT_USES_RAM) && ENABLED(SDSORT_CACHE_NAMES) && DISABLED(SDSORT_DYNAMIC_RAM)
|
|
|
|
#define SORTED_LONGNAME_MAXLEN ((SDSORT_CACHE_VFATS) * (FILENAME_LENGTH) + 1)
|
|
|
|
#else
|
|
|
|
#define SORTED_LONGNAME_MAXLEN LONG_FILENAME_LENGTH
|
|
|
|
#endif
|
|
|
|
|
2017-02-09 13:02:25 +00:00
|
|
|
// Cache filenames to speed up SD menus.
|
|
|
|
#if ENABLED(SDSORT_USES_RAM)
|
|
|
|
|
|
|
|
// If using dynamic ram for names, allocate on the heap.
|
|
|
|
#if ENABLED(SDSORT_CACHE_NAMES)
|
2017-02-09 13:05:34 +00:00
|
|
|
#if ENABLED(SDSORT_DYNAMIC_RAM)
|
|
|
|
char **sortshort, **sortnames;
|
|
|
|
#else
|
|
|
|
char sortshort[SDSORT_LIMIT][FILENAME_LENGTH];
|
2017-10-19 03:18:51 +00:00
|
|
|
char sortnames[SDSORT_LIMIT][SORTED_LONGNAME_MAXLEN];
|
2017-02-09 13:05:34 +00:00
|
|
|
#endif
|
2017-02-09 13:02:25 +00:00
|
|
|
#elif DISABLED(SDSORT_USES_STACK)
|
2017-10-19 03:18:51 +00:00
|
|
|
char sortnames[SDSORT_LIMIT][SORTED_LONGNAME_MAXLEN];
|
2017-02-09 13:02:25 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
// Folder sorting uses an isDir array when caching items.
|
2017-02-09 13:05:34 +00:00
|
|
|
#if HAS_FOLDER_SORTING
|
|
|
|
#if ENABLED(SDSORT_DYNAMIC_RAM)
|
|
|
|
uint8_t *isDir;
|
|
|
|
#elif ENABLED(SDSORT_CACHE_NAMES) || DISABLED(SDSORT_USES_STACK)
|
|
|
|
uint8_t isDir[(SDSORT_LIMIT+7)>>3];
|
|
|
|
#endif
|
2017-02-09 13:02:25 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#endif // SDSORT_USES_RAM
|
|
|
|
|
|
|
|
#endif // SDCARD_SORT_ALPHA
|
|
|
|
|
2011-11-06 21:48:15 +00:00
|
|
|
Sd2Card card;
|
|
|
|
SdVolume volume;
|
2011-11-06 20:39:53 +00:00
|
|
|
SdFile file;
|
2016-03-29 10:15:01 +00:00
|
|
|
|
2018-04-22 00:41:26 +00:00
|
|
|
#if ENABLED(POWER_LOSS_RECOVERY)
|
|
|
|
SdFile jobRecoveryFile;
|
|
|
|
#endif
|
|
|
|
|
2013-10-22 08:02:18 +00:00
|
|
|
#define SD_PROCEDURE_DEPTH 1
|
2015-03-02 15:06:01 +00:00
|
|
|
#define MAXPATHNAMELENGTH (FILENAME_LENGTH*MAX_DIR_DEPTH + MAX_DIR_DEPTH + 1)
|
2013-10-22 08:02:18 +00:00
|
|
|
uint8_t file_subcall_ctr;
|
|
|
|
uint32_t filespos[SD_PROCEDURE_DEPTH];
|
2016-03-29 10:15:01 +00:00
|
|
|
char proc_filenames[SD_PROCEDURE_DEPTH][MAXPATHNAMELENGTH];
|
2017-11-15 06:06:20 +00:00
|
|
|
uint32_t filesize, sdpos;
|
2011-11-06 20:39:53 +00:00
|
|
|
|
2016-03-29 10:19:27 +00:00
|
|
|
millis_t next_autostart_ms;
|
2011-11-06 21:48:15 +00:00
|
|
|
bool autostart_stilltocheck; //the sd start is delayed, because otherwise the serial cannot answer fast enought to make contact with the hostsoftware.
|
2015-03-02 15:06:01 +00:00
|
|
|
|
2011-11-19 12:13:34 +00:00
|
|
|
LsAction lsAction; //stored for recursion.
|
2015-03-02 15:06:01 +00:00
|
|
|
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.
|
2011-11-19 12:13:34 +00:00
|
|
|
char* diveDirName;
|
2018-01-21 23:13:56 +00:00
|
|
|
void lsDive(const char *prepend, SdFile parent, const char * const match=NULL
|
2017-11-05 14:49:38 +00:00
|
|
|
#if NUM_SERIAL > 1
|
|
|
|
, const int8_t port = -1
|
|
|
|
#endif
|
|
|
|
);
|
2017-02-09 13:02:25 +00:00
|
|
|
|
|
|
|
#if ENABLED(SDCARD_SORT_ALPHA)
|
|
|
|
void flush_presort();
|
|
|
|
#endif
|
2018-02-26 21:38:27 +00:00
|
|
|
|
|
|
|
#if ENABLED(AUTO_REPORT_SD_STATUS)
|
|
|
|
static uint8_t auto_report_sd_interval;
|
|
|
|
static millis_t next_sd_report_ms;
|
|
|
|
#if NUM_SERIAL > 1
|
|
|
|
static int8_t serialport;
|
|
|
|
#endif
|
|
|
|
#endif
|
2011-11-06 20:39:53 +00:00
|
|
|
};
|
2015-03-02 15:06:01 +00:00
|
|
|
|
2015-07-31 23:40:50 +00:00
|
|
|
#if PIN_EXISTS(SD_DETECT)
|
|
|
|
#if ENABLED(SD_DETECT_INVERTED)
|
2017-09-06 11:28:32 +00:00
|
|
|
#define IS_SD_INSERTED (READ(SD_DETECT_PIN) == HIGH)
|
2015-03-02 15:06:01 +00:00
|
|
|
#else
|
2017-09-06 11:28:32 +00:00
|
|
|
#define IS_SD_INSERTED (READ(SD_DETECT_PIN) == LOW)
|
2015-03-02 15:06:01 +00:00
|
|
|
#endif
|
2012-12-03 11:52:00 +00:00
|
|
|
#else
|
2018-03-07 07:42:11 +00:00
|
|
|
// No card detect line? Assume the card is inserted.
|
2015-03-02 15:06:01 +00:00
|
|
|
#define IS_SD_INSERTED true
|
2012-12-03 11:52:00 +00:00
|
|
|
#endif
|
|
|
|
|
2018-03-07 07:42:11 +00:00
|
|
|
extern CardReader card;
|
|
|
|
|
|
|
|
#endif // SDSUPPORT
|
|
|
|
|
2017-09-06 11:28:32 +00:00
|
|
|
#if ENABLED(SDSUPPORT)
|
|
|
|
#define IS_SD_PRINTING (card.sdprinting)
|
|
|
|
#define IS_SD_FILE_OPEN (card.isFileOpen())
|
2011-11-07 21:33:13 +00:00
|
|
|
#else
|
2017-09-06 11:28:32 +00:00
|
|
|
#define IS_SD_PRINTING (false)
|
|
|
|
#define IS_SD_FILE_OPEN (false)
|
|
|
|
#endif
|
2011-11-19 19:18:54 +00:00
|
|
|
|
2017-11-15 06:06:20 +00:00
|
|
|
#endif // _CARDREADER_H_
|