cardreader.cpp 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059
  1. #include "Marlin.h"
  2. #include "cardreader.h"
  3. #include "ultralcd.h"
  4. #include "stepper.h"
  5. #include "temperature.h"
  6. #include "language.h"
  7. #ifdef SDSUPPORT
  8. #define LONGEST_FILENAME (longFilename[0] ? longFilename : filename)
  9. CardReader::CardReader()
  10. {
  11. #ifdef SDCARD_SORT_ALPHA
  12. sort_count = 0;
  13. #if SDSORT_GCODE
  14. sort_alpha = true;
  15. sort_folders = FOLDER_SORTING;
  16. //sort_reverse = false;
  17. #endif
  18. #endif
  19. filesize = 0;
  20. sdpos = 0;
  21. sdprinting = false;
  22. cardOK = false;
  23. saving = false;
  24. logging = false;
  25. autostart_atmillis=0;
  26. workDirDepth = 0;
  27. file_subcall_ctr=0;
  28. memset(workDirParents, 0, sizeof(workDirParents));
  29. autostart_stilltocheck=true; //the SD start is delayed, because otherwise the serial cannot answer fast enough to make contact with the host software.
  30. lastnr=0;
  31. //power to SD reader
  32. #if SDPOWER > -1
  33. SET_OUTPUT(SDPOWER);
  34. WRITE(SDPOWER,HIGH);
  35. #endif //SDPOWER
  36. autostart_atmillis=millis()+5000;
  37. }
  38. char *createFilename(char *buffer,const dir_t &p) //buffer>12characters
  39. {
  40. char *pos=buffer;
  41. for (uint8_t i = 0; i < 11; i++)
  42. {
  43. if (p.name[i] == ' ')continue;
  44. if (i == 8)
  45. {
  46. *pos++='.';
  47. }
  48. *pos++=p.name[i];
  49. }
  50. *pos++=0;
  51. return buffer;
  52. }
  53. /**
  54. * Dive into a folder and recurse depth-first to perform a pre-set operation lsAction:
  55. * LS_Count - Add +1 to nrFiles for every file within the parent
  56. * LS_GetFilename - Get the filename of the file indexed by nrFiles
  57. * LS_SerialPrint - Print the full path and size of each file to serial output
  58. */
  59. void CardReader::lsDive(const char *prepend, SdFile parent, const char * const match/*=NULL*/) {
  60. dir_t p;
  61. uint8_t cnt = 0;
  62. // Read the next entry from a directory
  63. while (parent.readDir(p, longFilename) > 0) {
  64. // If the entry is a directory and the action is LS_SerialPrint
  65. if (DIR_IS_SUBDIR(&p) && lsAction != LS_Count && lsAction != LS_GetFilename) {
  66. // Get the short name for the item, which we know is a folder
  67. char lfilename[FILENAME_LENGTH];
  68. createFilename(lfilename, p);
  69. // Allocate enough stack space for the full path to a folder, trailing slash, and nul
  70. bool prepend_is_empty = (prepend[0] == '\0');
  71. int len = (prepend_is_empty ? 1 : strlen(prepend)) + strlen(lfilename) + 1 + 1;
  72. char path[len];
  73. // Append the FOLDERNAME12/ to the passed string.
  74. // It contains the full path to the "parent" argument.
  75. // We now have the full path to the item in this folder.
  76. strcpy(path, prepend_is_empty ? "/" : prepend); // root slash if prepend is empty
  77. strcat(path, lfilename); // FILENAME_LENGTH-1 characters maximum
  78. strcat(path, "/"); // 1 character
  79. // Serial.print(path);
  80. // Get a new directory object using the full path
  81. // and dive recursively into it.
  82. SdFile dir;
  83. if (!dir.open(parent, lfilename, O_READ)) {
  84. if (lsAction == LS_SerialPrint) {
  85. //SERIAL_ECHO_START();
  86. //SERIAL_ECHOPGM(MSG_SD_CANT_OPEN_SUBDIR);
  87. //SERIAL_ECHOLN(lfilename);
  88. }
  89. }
  90. lsDive(path, dir);
  91. // close() is done automatically by destructor of SdFile
  92. }
  93. else {
  94. uint8_t pn0 = p.name[0];
  95. if (pn0 == DIR_NAME_FREE) break;
  96. if (pn0 == DIR_NAME_DELETED || pn0 == '.') continue;
  97. if (longFilename[0] == '.') continue;
  98. if (!DIR_IS_FILE_OR_SUBDIR(&p) || (p.attributes & DIR_ATT_HIDDEN)) continue;
  99. filenameIsDir = DIR_IS_SUBDIR(&p);
  100. if (!filenameIsDir && (p.name[8] != 'G' || p.name[9] == '~')) continue;
  101. switch (lsAction) {
  102. case LS_Count:
  103. nrFiles++;
  104. break;
  105. case LS_SerialPrint:
  106. createFilename(filename, p);
  107. SERIAL_PROTOCOL(prepend);
  108. SERIAL_PROTOCOL(filename);
  109. MYSERIAL.write(' ');
  110. SERIAL_PROTOCOLLN(p.fileSize);
  111. break;
  112. case LS_GetFilename:
  113. createFilename(filename, p);
  114. cluster = parent.curCluster();
  115. position = parent.curPosition();
  116. creationDate = p.creationDate;
  117. creationTime = p.creationTime;
  118. if (match != NULL) {
  119. if (strcasecmp(match, filename) == 0) return;
  120. }
  121. else if (cnt == nrFiles) {
  122. return;
  123. }
  124. cnt++;
  125. break;
  126. }
  127. }
  128. } // while readDir
  129. }
  130. void CardReader::ls()
  131. {
  132. lsAction=LS_SerialPrint;
  133. // if(lsAction==LS_Count)
  134. // nrFiles=0;
  135. root.rewind();
  136. lsDive("",root);
  137. }
  138. void CardReader::initsd()
  139. {
  140. cardOK = false;
  141. if(root.isOpen())
  142. root.close();
  143. #ifdef SDSLOW
  144. if (!card.init(SPI_HALF_SPEED,SDSS)
  145. #if defined(LCD_SDSS) && (LCD_SDSS != SDSS)
  146. && !card.init(SPI_HALF_SPEED,LCD_SDSS)
  147. #endif
  148. )
  149. #else
  150. if (!card.init(SPI_FULL_SPEED,SDSS)
  151. #if defined(LCD_SDSS) && (LCD_SDSS != SDSS)
  152. && !card.init(SPI_FULL_SPEED,LCD_SDSS)
  153. #endif
  154. )
  155. #endif
  156. {
  157. //if (!card.init(SPI_HALF_SPEED,SDSS))
  158. SERIAL_ECHO_START;
  159. SERIAL_ECHOLNRPGM(MSG_SD_INIT_FAIL);
  160. }
  161. else if (!volume.init(&card))
  162. {
  163. SERIAL_ERROR_START;
  164. SERIAL_ERRORLNRPGM(MSG_SD_VOL_INIT_FAIL);
  165. }
  166. else if (!root.openRoot(&volume))
  167. {
  168. SERIAL_ERROR_START;
  169. SERIAL_ERRORLNRPGM(MSG_SD_OPENROOT_FAIL);
  170. }
  171. else
  172. {
  173. cardOK = true;
  174. SERIAL_ECHO_START;
  175. SERIAL_ECHOLNRPGM(MSG_SD_CARD_OK);
  176. }
  177. workDir=root;
  178. curDir=&root;
  179. #ifdef SDCARD_SORT_ALPHA
  180. presort();
  181. #endif
  182. /*
  183. if(!workDir.openRoot(&volume))
  184. {
  185. SERIAL_ECHOLNPGM(MSG_SD_WORKDIR_FAIL);
  186. }
  187. */
  188. }
  189. void CardReader::setroot()
  190. {
  191. /*if(!workDir.openRoot(&volume))
  192. {
  193. SERIAL_ECHOLNPGM(MSG_SD_WORKDIR_FAIL);
  194. }*/
  195. workDir=root;
  196. curDir=&workDir;
  197. #ifdef SDCARD_SORT_ALPHA
  198. presort();
  199. #endif
  200. }
  201. void CardReader::release()
  202. {
  203. sdprinting = false;
  204. cardOK = false;
  205. }
  206. void CardReader::startFileprint()
  207. {
  208. if(cardOK)
  209. {
  210. sdprinting = true;
  211. #ifdef SDCARD_SORT_ALPHA
  212. // flush_presort();
  213. #endif
  214. }
  215. }
  216. void CardReader::pauseSDPrint()
  217. {
  218. if(sdprinting)
  219. {
  220. sdprinting = false;
  221. }
  222. }
  223. void CardReader::openLogFile(char* name)
  224. {
  225. logging = true;
  226. openFile(name, false);
  227. }
  228. void CardReader::getAbsFilename(char *t)
  229. {
  230. uint8_t cnt=0;
  231. *t='/';t++;cnt++;
  232. for(uint8_t i=0;i<workDirDepth;i++)
  233. {
  234. workDirParents[i].getFilename(t); //SDBaseFile.getfilename!
  235. while(*t!=0 && cnt< MAXPATHNAMELENGTH)
  236. {t++;cnt++;} //crawl counter forward.
  237. }
  238. if(cnt<MAXPATHNAMELENGTH-13)
  239. file.getFilename(t);
  240. else
  241. t[0]=0;
  242. }
  243. void CardReader::openFile(char* name,bool read, bool replace_current/*=true*/)
  244. {
  245. if(!cardOK)
  246. return;
  247. if(file.isOpen()) //replacing current file by new file, or subfile call
  248. {
  249. if(!replace_current)
  250. {
  251. if((int)file_subcall_ctr>(int)SD_PROCEDURE_DEPTH-1)
  252. {
  253. SERIAL_ERROR_START;
  254. SERIAL_ERRORPGM("trying to call sub-gcode files with too many levels. MAX level is:");
  255. SERIAL_ERRORLN(SD_PROCEDURE_DEPTH);
  256. kill();
  257. return;
  258. }
  259. SERIAL_ECHO_START;
  260. SERIAL_ECHOPGM("SUBROUTINE CALL target:\"");
  261. SERIAL_ECHO(name);
  262. SERIAL_ECHOPGM("\" parent:\"");
  263. //store current filename and position
  264. getAbsFilename(filenames[file_subcall_ctr]);
  265. SERIAL_ECHO(filenames[file_subcall_ctr]);
  266. SERIAL_ECHOPGM("\" pos");
  267. SERIAL_ECHOLN(sdpos);
  268. filespos[file_subcall_ctr]=sdpos;
  269. file_subcall_ctr++;
  270. }
  271. else
  272. {
  273. SERIAL_ECHO_START;
  274. SERIAL_ECHOPGM("Now doing file: ");
  275. SERIAL_ECHOLN(name);
  276. }
  277. file.close();
  278. }
  279. else //opening fresh file
  280. {
  281. file_subcall_ctr=0; //resetting procedure depth in case user cancels print while in procedure
  282. SERIAL_ECHO_START;
  283. SERIAL_ECHOPGM("Now fresh file: ");
  284. SERIAL_ECHOLN(name);
  285. }
  286. sdprinting = false;
  287. SdFile myDir;
  288. curDir=&root;
  289. char *fname=name;
  290. char *dirname_start,*dirname_end;
  291. if(name[0]=='/')
  292. {
  293. dirname_start=strchr(name,'/')+1;
  294. while(dirname_start)
  295. {
  296. dirname_end=strchr(dirname_start,'/');
  297. //SERIAL_ECHO("start:");SERIAL_ECHOLN((int)(dirname_start-name));
  298. //SERIAL_ECHO("end :");SERIAL_ECHOLN((int)(dirname_end-name));
  299. if(dirname_end && dirname_end>dirname_start)
  300. {
  301. char subdirname[13];
  302. strncpy(subdirname, dirname_start, dirname_end-dirname_start);
  303. subdirname[dirname_end-dirname_start]=0;
  304. SERIAL_ECHOLN(subdirname);
  305. if(!myDir.open(curDir,subdirname,O_READ))
  306. {
  307. SERIAL_PROTOCOLRPGM(MSG_SD_OPEN_FILE_FAIL);
  308. SERIAL_PROTOCOL(subdirname);
  309. SERIAL_PROTOCOLLNPGM(".");
  310. return;
  311. }
  312. else
  313. {
  314. //SERIAL_ECHOLN("dive ok");
  315. }
  316. curDir=&myDir;
  317. dirname_start=dirname_end+1;
  318. }
  319. else // the reminder after all /fsa/fdsa/ is the filename
  320. {
  321. fname=dirname_start;
  322. //SERIAL_ECHOLN("remaider");
  323. //SERIAL_ECHOLN(fname);
  324. break;
  325. }
  326. }
  327. }
  328. else //relative path
  329. {
  330. curDir=&workDir;
  331. }
  332. if(read)
  333. {
  334. if (file.open(curDir, fname, O_READ))
  335. {
  336. filesize = file.fileSize();
  337. SERIAL_PROTOCOLRPGM(MSG_SD_FILE_OPENED);
  338. SERIAL_PROTOCOL(fname);
  339. SERIAL_PROTOCOLRPGM(MSG_SD_SIZE);
  340. SERIAL_PROTOCOLLN(filesize);
  341. sdpos = 0;
  342. SERIAL_PROTOCOLLNRPGM(MSG_SD_FILE_SELECTED);
  343. getfilename(0, fname);
  344. lcd_setstatus(longFilename[0] ? longFilename : fname);
  345. lcd_setstatus("SD-PRINTING ");
  346. }
  347. else
  348. {
  349. SERIAL_PROTOCOLRPGM(MSG_SD_OPEN_FILE_FAIL);
  350. SERIAL_PROTOCOL(fname);
  351. SERIAL_PROTOCOLLNPGM(".");
  352. }
  353. }
  354. else
  355. { //write
  356. if (!file.open(curDir, fname, O_CREAT | O_APPEND | O_WRITE | O_TRUNC))
  357. {
  358. SERIAL_PROTOCOLRPGM(MSG_SD_OPEN_FILE_FAIL);
  359. SERIAL_PROTOCOL(fname);
  360. SERIAL_PROTOCOLLNPGM(".");
  361. }
  362. else
  363. {
  364. saving = true;
  365. SERIAL_PROTOCOLRPGM(MSG_SD_WRITE_TO_FILE);
  366. SERIAL_PROTOCOLLN(name);
  367. lcd_setstatus(fname);
  368. }
  369. }
  370. }
  371. void CardReader::removeFile(char* name)
  372. {
  373. if(!cardOK)
  374. return;
  375. file.close();
  376. sdprinting = false;
  377. SdFile myDir;
  378. curDir=&root;
  379. char *fname=name;
  380. char *dirname_start,*dirname_end;
  381. if(name[0]=='/')
  382. {
  383. dirname_start=strchr(name,'/')+1;
  384. while(dirname_start)
  385. {
  386. dirname_end=strchr(dirname_start,'/');
  387. //SERIAL_ECHO("start:");SERIAL_ECHOLN((int)(dirname_start-name));
  388. //SERIAL_ECHO("end :");SERIAL_ECHOLN((int)(dirname_end-name));
  389. if(dirname_end && dirname_end>dirname_start)
  390. {
  391. char subdirname[13];
  392. strncpy(subdirname, dirname_start, dirname_end-dirname_start);
  393. subdirname[dirname_end-dirname_start]=0;
  394. SERIAL_ECHOLN(subdirname);
  395. if(!myDir.open(curDir,subdirname,O_READ))
  396. {
  397. SERIAL_PROTOCOLRPGM("open failed, File: ");
  398. SERIAL_PROTOCOL(subdirname);
  399. SERIAL_PROTOCOLLNPGM(".");
  400. return;
  401. }
  402. else
  403. {
  404. //SERIAL_ECHOLN("dive ok");
  405. }
  406. curDir=&myDir;
  407. dirname_start=dirname_end+1;
  408. }
  409. else // the reminder after all /fsa/fdsa/ is the filename
  410. {
  411. fname=dirname_start;
  412. //SERIAL_ECHOLN("remaider");
  413. //SERIAL_ECHOLN(fname);
  414. break;
  415. }
  416. }
  417. }
  418. else //relative path
  419. {
  420. curDir=&workDir;
  421. }
  422. if (file.remove(curDir, fname))
  423. {
  424. SERIAL_PROTOCOLPGM("File deleted:");
  425. SERIAL_PROTOCOLLN(fname);
  426. sdpos = 0;
  427. #ifdef SDCARD_SORT_ALPHA
  428. presort();
  429. #endif
  430. }
  431. else
  432. {
  433. SERIAL_PROTOCOLPGM("Deletion failed, File: ");
  434. SERIAL_PROTOCOL(fname);
  435. SERIAL_PROTOCOLLNPGM(".");
  436. }
  437. }
  438. uint32_t CardReader::getFileSize()
  439. {
  440. return filesize;
  441. }
  442. void CardReader::getStatus()
  443. {
  444. if(sdprinting){
  445. SERIAL_PROTOCOL(longFilename);
  446. SERIAL_PROTOCOLPGM("\n");
  447. SERIAL_PROTOCOLRPGM(MSG_SD_PRINTING_BYTE);
  448. SERIAL_PROTOCOL(sdpos);
  449. SERIAL_PROTOCOLPGM("/");
  450. SERIAL_PROTOCOLLN(filesize);
  451. uint16_t time = millis()/60000 - starttime/60000;
  452. SERIAL_PROTOCOL(itostr2(time/60));
  453. SERIAL_PROTOCOL(':');
  454. SERIAL_PROTOCOL(itostr2(time%60));
  455. SERIAL_PROTOCOLPGM("\n");
  456. }
  457. else{
  458. SERIAL_PROTOCOLLNRPGM("Not printing");
  459. }
  460. }
  461. void CardReader::write_command(char *buf)
  462. {
  463. char* begin = buf;
  464. char* npos = 0;
  465. char* end = buf + strlen(buf) - 1;
  466. file.writeError = false;
  467. if((npos = strchr(buf, 'N')) != NULL)
  468. {
  469. begin = strchr(npos, ' ') + 1;
  470. end = strchr(npos, '*') - 1;
  471. }
  472. end[1] = '\r';
  473. end[2] = '\n';
  474. end[3] = '\0';
  475. file.write(begin);
  476. if (file.writeError)
  477. {
  478. SERIAL_ERROR_START;
  479. SERIAL_ERRORLNRPGM(MSG_SD_ERR_WRITE_TO_FILE);
  480. }
  481. }
  482. #define CHUNK_SIZE 64
  483. void CardReader::write_command_no_newline(char *buf)
  484. {
  485. file.write(buf, CHUNK_SIZE);
  486. if (file.writeError)
  487. {
  488. SERIAL_ERROR_START;
  489. SERIAL_ERRORLNRPGM(MSG_SD_ERR_WRITE_TO_FILE);
  490. MYSERIAL.println("An error while writing to the SD Card.");
  491. }
  492. }
  493. void CardReader::checkautostart(bool force)
  494. {
  495. if(!force)
  496. {
  497. if(!autostart_stilltocheck)
  498. return;
  499. if(autostart_atmillis<millis())
  500. return;
  501. }
  502. autostart_stilltocheck=false;
  503. if(!cardOK)
  504. {
  505. initsd();
  506. if(!cardOK) //fail
  507. return;
  508. }
  509. char autoname[30];
  510. sprintf_P(autoname, PSTR("auto%i.g"), lastnr);
  511. for(int8_t i=0;i<(int8_t)strlen(autoname);i++)
  512. autoname[i]=tolower(autoname[i]);
  513. dir_t p;
  514. root.rewind();
  515. bool found=false;
  516. while (root.readDir(p, NULL) > 0)
  517. {
  518. for(int8_t i=0;i<(int8_t)strlen((char*)p.name);i++)
  519. p.name[i]=tolower(p.name[i]);
  520. //Serial.print((char*)p.name);
  521. //Serial.print(" ");
  522. //Serial.println(autoname);
  523. if(p.name[9]!='~') //skip safety copies
  524. if(strncmp((char*)p.name,autoname,5)==0)
  525. {
  526. char cmd[30];
  527. // M23: Select SD file
  528. sprintf_P(cmd, PSTR("M23 %s"), autoname);
  529. enquecommand(cmd);
  530. // M24: Start/resume SD print
  531. enquecommand_P(PSTR("M24"));
  532. found=true;
  533. }
  534. }
  535. if(!found)
  536. lastnr=-1;
  537. else
  538. lastnr++;
  539. }
  540. void CardReader::closefile(bool store_location)
  541. {
  542. file.sync();
  543. file.close();
  544. saving = false;
  545. logging = false;
  546. if(store_location)
  547. {
  548. //future: store printer state, filename and position for continuing a stopped print
  549. // so one can unplug the printer and continue printing the next day.
  550. }
  551. }
  552. void CardReader::getfilename(uint16_t nr, const char * const match/*=NULL*/)
  553. {
  554. curDir=&workDir;
  555. lsAction=LS_GetFilename;
  556. nrFiles=nr;
  557. curDir->rewind();
  558. lsDive("",*curDir,match);
  559. }
  560. void CardReader::getfilename_simple(uint32_t position, const char * const match/*=NULL*/)
  561. {
  562. curDir = &workDir;
  563. lsAction = LS_GetFilename;
  564. nrFiles = 0;
  565. curDir->seekSet(position);
  566. lsDive("", *curDir, match);
  567. }
  568. uint16_t CardReader::getnrfilenames()
  569. {
  570. curDir=&workDir;
  571. lsAction=LS_Count;
  572. nrFiles=0;
  573. curDir->rewind();
  574. lsDive("",*curDir);
  575. //SERIAL_ECHOLN(nrFiles);
  576. return nrFiles;
  577. }
  578. void CardReader::chdir(const char * relpath)
  579. {
  580. SdFile newfile;
  581. SdFile *parent=&root;
  582. if(workDir.isOpen())
  583. parent=&workDir;
  584. if(!newfile.open(*parent,relpath, O_READ))
  585. {
  586. SERIAL_ECHO_START;
  587. SERIAL_ECHORPGM(MSG_SD_CANT_ENTER_SUBDIR);
  588. SERIAL_ECHOLN(relpath);
  589. }
  590. else
  591. {
  592. if (workDirDepth < MAX_DIR_DEPTH) {
  593. for (int d = ++workDirDepth; d--;)
  594. workDirParents[d+1] = workDirParents[d];
  595. workDirParents[0]=*parent;
  596. }
  597. workDir=newfile;
  598. #ifdef SDCARD_SORT_ALPHA
  599. presort();
  600. #endif
  601. }
  602. }
  603. void CardReader::updir()
  604. {
  605. if(workDirDepth > 0)
  606. {
  607. --workDirDepth;
  608. workDir = workDirParents[0];
  609. for (uint8_t d = 0; d < workDirDepth; d++)
  610. workDirParents[d] = workDirParents[d+1];
  611. #ifdef SDCARD_SORT_ALPHA
  612. presort();
  613. #endif
  614. }
  615. }
  616. #ifdef SDCARD_SORT_ALPHA
  617. /**
  618. * Get the name of a file in the current directory by sort-index
  619. */
  620. void CardReader::getfilename_sorted(const uint16_t nr) {
  621. getfilename(
  622. #if SDSORT_GCODE
  623. sort_alpha &&
  624. #endif
  625. (nr < sort_count) ? sort_order[nr] : nr
  626. );
  627. }
  628. #ifdef SDSORT_QUICKSORT
  629. void CardReader::swap(uint8_t left, uint8_t right) {
  630. uint8_t tmp = sort_order[right];
  631. sort_order[right] = sort_order[left];
  632. sort_order[left] = tmp;
  633. }
  634. void CardReader::quicksort(uint8_t left, uint8_t right) {
  635. if (left < right) {
  636. char name_left[LONG_FILENAME_LENGTH + 1];
  637. char name_i[LONG_FILENAME_LENGTH + 1];
  638. uint16_t creation_time_left;
  639. uint16_t creation_date_left;
  640. uint8_t boundary = left;
  641. for (uint8_t i = left+1; i < right; i++) {
  642. uint8_t o_left = sort_order[left];
  643. uint8_t o_i = sort_order[i];
  644. getfilename_simple(positions[o_left]);
  645. strcpy(name_left, LONGEST_FILENAME); // save (or getfilename below will trounce it)
  646. creation_date_left = creationDate;
  647. creation_time_left = creationTime;
  648. getfilename_simple(positions[o_i]);
  649. strcpy(name_i, LONGEST_FILENAME);
  650. if (strcasecmp(name_left, name_i) > 0) {
  651. swap(i, ++boundary);
  652. }
  653. }
  654. swap(left, boundary);
  655. quicksort(left, boundary);
  656. quicksort(boundary + 1, right);
  657. }
  658. }
  659. #endif //SDSORT_QUICKSORT
  660. /**
  661. * Read all the files and produce a sort key
  662. *
  663. * We can do this in 3 ways...
  664. * - Minimal RAM: Read two filenames at a time sorting along...
  665. * - Some RAM: Buffer the directory just for this sort
  666. * - Most RAM: Buffer the directory and return filenames from RAM
  667. */
  668. void CardReader::presort() {
  669. if (farm_mode || IS_SD_INSERTED == false) return; //sorting is not used in farm mode
  670. uint8_t sdSort = eeprom_read_byte((uint8_t*)EEPROM_SD_SORT);
  671. if (sdSort == SD_SORT_NONE) return; //sd sort is turned off
  672. #if SDSORT_GCODE
  673. if (!sort_alpha) return;
  674. #endif
  675. KEEPALIVE_STATE(IN_HANDLER);
  676. // Throw away old sort index
  677. flush_presort();
  678. // If there are files, sort up to the limit
  679. uint16_t fileCnt = getnrfilenames();
  680. if (fileCnt > 0) {
  681. // Never sort more than the max allowed
  682. // If you use folders to organize, 20 may be enough
  683. if (fileCnt > SDSORT_LIMIT) {
  684. lcd_show_fullscreen_message_and_wait_P(MSG_FILE_CNT);
  685. fileCnt = SDSORT_LIMIT;
  686. }
  687. lcd_implementation_clear();
  688. #if !SDSORT_USES_RAM
  689. lcd_set_progress();
  690. #endif
  691. lcd_print_at_PGM(0, 1, MSG_SORTING);
  692. // Sort order is always needed. May be static or dynamic.
  693. #if SDSORT_DYNAMIC_RAM
  694. sort_order = new uint8_t[fileCnt];
  695. #endif
  696. // Use RAM to store the entire directory during pre-sort.
  697. // SDSORT_LIMIT should be set to prevent over-allocation.
  698. #if SDSORT_USES_RAM
  699. // If using dynamic ram for names, allocate on the heap.
  700. #if SDSORT_CACHE_NAMES
  701. #if SDSORT_DYNAMIC_RAM
  702. sortshort = new char*[fileCnt];
  703. sortnames = new char*[fileCnt];
  704. #endif
  705. #elif SDSORT_USES_STACK
  706. char sortnames[fileCnt][LONG_FILENAME_LENGTH];
  707. uint16_t creation_time[fileCnt];
  708. uint16_t creation_date[fileCnt];
  709. #endif
  710. // Folder sorting needs 1 bit per entry for flags.
  711. #if HAS_FOLDER_SORTING
  712. #if SDSORT_DYNAMIC_RAM
  713. isDir = new uint8_t[(fileCnt + 7) >> 3];
  714. #elif SDSORT_USES_STACK
  715. uint8_t isDir[(fileCnt + 7) >> 3];
  716. #endif
  717. #endif
  718. #else // !SDSORT_USES_RAM
  719. uint32_t positions[fileCnt];
  720. // By default re-read the names from SD for every compare
  721. // retaining only two filenames at a time. This is very
  722. // slow but is safest and uses minimal RAM.
  723. char name1[LONG_FILENAME_LENGTH + 1];
  724. uint16_t creation_time_bckp;
  725. uint16_t creation_date_bckp;
  726. #endif
  727. position = 0;
  728. if (fileCnt > 1) {
  729. // Init sort order.
  730. for (uint16_t i = 0; i < fileCnt; i++) {
  731. if (!IS_SD_INSERTED) return;
  732. manage_heater();
  733. sort_order[i] = i;
  734. positions[i] = position;
  735. getfilename(i);
  736. // If using RAM then read all filenames now.
  737. #if SDSORT_USES_RAM
  738. getfilename(i);
  739. #if SDSORT_DYNAMIC_RAM
  740. // Use dynamic method to copy long filename
  741. sortnames[i] = strdup(LONGEST_FILENAME);
  742. #if SDSORT_CACHE_NAMES
  743. // When caching also store the short name, since
  744. // we're replacing the getfilename() behavior.
  745. sortshort[i] = strdup(filename);
  746. #endif
  747. #else
  748. // Copy filenames into the static array
  749. strcpy(sortnames[i], LONGEST_FILENAME);
  750. creation_time[i] = creationTime;
  751. creation_date[i] = creationDate;
  752. #if SDSORT_CACHE_NAMES
  753. strcpy(sortshort[i], filename);
  754. #endif
  755. #endif
  756. // char out[30];
  757. // sprintf_P(out, PSTR("---- %i %s %s"), i, filenameIsDir ? "D" : " ", sortnames[i]);
  758. // SERIAL_ECHOLN(out);
  759. #if HAS_FOLDER_SORTING
  760. const uint16_t bit = i & 0x07, ind = i >> 3;
  761. if (bit == 0) isDir[ind] = 0x00;
  762. if (filenameIsDir) isDir[ind] |= _BV(bit);
  763. #endif
  764. #endif
  765. }
  766. #ifdef QUICKSORT
  767. quicksort(0, fileCnt - 1);
  768. #else //Qicksort not defined, use Bubble Sort
  769. uint32_t counter = 0;
  770. uint16_t total = 0.5*(fileCnt-1)*(fileCnt);
  771. // Compare names from the array or just the two buffered names
  772. #if SDSORT_USES_RAM
  773. #define _SORT_CMP_NODIR() (strcasecmp(sortnames[o1], sortnames[o2]) > 0)
  774. #define _SORT_CMP_TIME_NODIR() (((creation_date[o1] == creation_date[o2]) && (creation_time[o1] < creation_time[o2])) || \
  775. (creation_date[o1] < creation_date [o2]))
  776. #else
  777. #define _SORT_CMP_NODIR() (strcasecmp(name1, name2) > 0) //true if lowercase(name1) > lowercase(name2)
  778. #define _SORT_CMP_TIME_NODIR() (((creation_date_bckp == creationDate) && (creation_time_bckp > creationTime)) || \
  779. (creation_date_bckp > creationDate))
  780. #endif
  781. #if HAS_FOLDER_SORTING
  782. #if SDSORT_USES_RAM
  783. // Folder sorting needs an index and bit to test for folder-ness.
  784. const uint8_t ind1 = o1 >> 3, bit1 = o1 & 0x07,
  785. ind2 = o2 >> 3, bit2 = o2 & 0x07;
  786. #define _SORT_CMP_DIR(fs) \
  787. (((isDir[ind1] & _BV(bit1)) != 0) == ((isDir[ind2] & _BV(bit2)) != 0) \
  788. ? _SORT_CMP_NODIR() \
  789. : (isDir[fs > 0 ? ind1 : ind2] & (fs > 0 ? _BV(bit1) : _BV(bit2))) != 0)
  790. #define _SORT_CMP_TIME_DIR(fs) \
  791. (((isDir[ind1] & _BV(bit1)) != 0) == ((isDir[ind2] & _BV(bit2)) != 0) \
  792. ? _SORT_CMP_TIME_NODIR() \
  793. : (isDir[fs > 0 ? ind1 : ind2] & (fs > 0 ? _BV(bit1) : _BV(bit2))) != 0)
  794. #else
  795. #define _SORT_CMP_DIR(fs) ((dir1 == filenameIsDir) ? _SORT_CMP_NODIR() : (fs > 0 ? dir1 : !dir1))
  796. #define _SORT_CMP_TIME_DIR(fs) ((dir1 == filenameIsDir) ? _SORT_CMP_TIME_NODIR() : (fs < 0 ? dir1 : !dir1))
  797. #endif
  798. #endif
  799. for (uint16_t i = fileCnt; --i;) {
  800. if (!IS_SD_INSERTED) return;
  801. bool didSwap = false;
  802. #if !SDSORT_USES_RAM //show progresss bar only if slow sorting method is used
  803. int8_t percent = (counter * 100) / total;//((counter * 100) / pow((fileCnt-1),2));
  804. for (int column = 0; column < 20; column++) {
  805. if (column < (percent/5)) lcd_implementation_print_at(column, 2, "\x01"); //simple progress bar
  806. }
  807. #endif
  808. //MYSERIAL.println(int(i));
  809. for (uint16_t j = 0; j < i; ++j) {
  810. if (!IS_SD_INSERTED) return;
  811. manage_heater();
  812. const uint16_t o1 = sort_order[j], o2 = sort_order[j + 1];
  813. // The most economical method reads names as-needed
  814. // throughout the loop. Slow if there are many.
  815. #if !SDSORT_USES_RAM
  816. counter++;
  817. getfilename_simple(positions[o1]);
  818. strcpy(name1, LONGEST_FILENAME); // save (or getfilename below will trounce it)
  819. creation_date_bckp = creationDate;
  820. creation_time_bckp = creationTime;
  821. #if HAS_FOLDER_SORTING
  822. bool dir1 = filenameIsDir;
  823. #endif
  824. getfilename_simple(positions[o2]);
  825. char *name2 = LONGEST_FILENAME; // use the string in-place
  826. #endif // !SDSORT_USES_RAM
  827. // Sort the current pair according to settings.
  828. if(
  829. #if HAS_FOLDER_SORTING
  830. (sdSort == SD_SORT_TIME && _SORT_CMP_TIME_DIR(FOLDER_SORTING)) || (sdSort == SD_SORT_ALPHA && _SORT_CMP_DIR(FOLDER_SORTING))
  831. #else
  832. (sdSort == SD_SORT_TIME && _SORT_CMP_TIME_NODIR()) || (sdSort == SD_SORT_ALPHA && _SORT_CMP_NODIR())
  833. #endif
  834. )
  835. {
  836. sort_order[j] = o2;
  837. sort_order[j + 1] = o1;
  838. didSwap = true;
  839. }
  840. }
  841. if (!didSwap) break;
  842. } //end of bubble sort loop
  843. #endif
  844. // Using RAM but not keeping names around
  845. #if (SDSORT_USES_RAM && !SDSORT_CACHE_NAMES)
  846. #if SDSORT_DYNAMIC_RAM
  847. for (uint16_t i = 0; i < fileCnt; ++i) free(sortnames[i]);
  848. #if HAS_FOLDER_SORTING
  849. free(isDir);
  850. #endif
  851. #endif
  852. #endif
  853. }
  854. else {
  855. sort_order[0] = 0;
  856. #if (SDSORT_USES_RAM && SDSORT_CACHE_NAMES)
  857. getfilename(0);
  858. #if SDSORT_DYNAMIC_RAM
  859. sortnames = new char*[1];
  860. sortnames[0] = strdup(LONGEST_FILENAME); // malloc
  861. sortshort = new char*[1];
  862. sortshort[0] = strdup(filename); // malloc
  863. isDir = new uint8_t[1];
  864. #else
  865. strcpy(sortnames[0], LONGEST_FILENAME);
  866. strcpy(sortshort[0], filename);
  867. #endif
  868. isDir[0] = filenameIsDir ? 0x01 : 0x00;
  869. #endif
  870. }
  871. sort_count = fileCnt;
  872. }
  873. #if !SDSORT_USES_RAM //show progress bar only if slow sorting method is used
  874. for (int column = 0; column <= 19; column++) lcd_implementation_print_at(column, 2, "\x01"); //simple progress bar
  875. delay(300);
  876. lcd_set_degree();
  877. lcd_implementation_clear();
  878. lcd_update(2);
  879. #endif
  880. lcd_update(2);
  881. KEEPALIVE_STATE(NOT_BUSY);
  882. lcd_timeoutToStatus = millis() + LCD_TIMEOUT_TO_STATUS;
  883. }
  884. void CardReader::flush_presort() {
  885. if (sort_count > 0) {
  886. #if SDSORT_DYNAMIC_RAM
  887. delete sort_order;
  888. #if SDSORT_CACHE_NAMES
  889. for (uint8_t i = 0; i < sort_count; ++i) {
  890. free(sortshort[i]); // strdup
  891. free(sortnames[i]); // strdup
  892. }
  893. delete sortshort;
  894. delete sortnames;
  895. #endif
  896. #endif
  897. sort_count = 0;
  898. }
  899. }
  900. #endif // SDCARD_SORT_ALPHA
  901. void CardReader::printingHasFinished()
  902. {
  903. st_synchronize();
  904. if(file_subcall_ctr>0) //heading up to a parent file that called current as a procedure.
  905. {
  906. file.close();
  907. file_subcall_ctr--;
  908. openFile(filenames[file_subcall_ctr],true,true);
  909. setIndex(filespos[file_subcall_ctr]);
  910. startFileprint();
  911. }
  912. else
  913. {
  914. quickStop();
  915. file.close();
  916. sdprinting = false;
  917. if(SD_FINISHED_STEPPERRELEASE)
  918. {
  919. finishAndDisableSteppers();
  920. //enquecommand_P(PSTR(SD_FINISHED_RELEASECOMMAND));
  921. }
  922. autotempShutdown();
  923. #ifdef SDCARD_SORT_ALPHA
  924. //if(!check_file) presort();
  925. #endif
  926. }
  927. }
  928. bool CardReader::ToshibaFlashAir_GetIP(uint8_t *ip)
  929. {
  930. memset(ip, 0, 4);
  931. return card.readExtMemory(1, 1, 0x400+0x150, 4, ip);
  932. }
  933. #endif //SDSUPPORT