My Marlin configs for Fabrikator Mini and CTC i3 Pro B
Du kan inte välja fler än 25 ämnen Ämnen måste starta med en bokstav eller siffra, kan innehålla bindestreck ('-') och vara max 35 tecken långa.

cardreader.cpp 10KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528
  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. CardReader::CardReader()
  9. {
  10. filesize = 0;
  11. sdpos = 0;
  12. sdprinting = false;
  13. cardOK = false;
  14. saving = false;
  15. autostart_atmillis=0;
  16. autostart_stilltocheck=true; //the sd start is delayed, because otherwise the serial cannot answer fast enought to make contact with the hostsoftware.
  17. lastnr=0;
  18. //power to SD reader
  19. #if SDPOWER > -1
  20. SET_OUTPUT(SDPOWER);
  21. WRITE(SDPOWER,HIGH);
  22. #endif //SDPOWER
  23. autostart_atmillis=millis()+5000;
  24. }
  25. char *createFilename(char *buffer,const dir_t &p) //buffer>12characters
  26. {
  27. char *pos=buffer;
  28. for (uint8_t i = 0; i < 11; i++)
  29. {
  30. if (p.name[i] == ' ')continue;
  31. if (i == 8)
  32. {
  33. *pos++='.';
  34. }
  35. *pos++=p.name[i];
  36. }
  37. *pos++=0;
  38. return buffer;
  39. }
  40. void CardReader::lsDive(const char *prepend,SdFile parent)
  41. {
  42. dir_t p;
  43. uint8_t cnt=0;
  44. while (parent.readDir(p) > 0)
  45. {
  46. if( DIR_IS_SUBDIR(&p) && lsAction!=LS_Count && lsAction!=LS_GetFilename)
  47. {
  48. char path[13*2];
  49. char lfilename[13];
  50. createFilename(lfilename,p);
  51. path[0]=0;
  52. if(strlen(prepend)==0) //avoid leading / if already in prepend
  53. {
  54. strcat(path,"/");
  55. }
  56. strcat(path,prepend);
  57. strcat(path,lfilename);
  58. strcat(path,"/");
  59. //Serial.print(path);
  60. SdFile dir;
  61. if(!dir.open(parent,lfilename, O_READ))
  62. {
  63. if(lsAction==LS_SerialPrint)
  64. {
  65. SERIAL_ECHO_START;
  66. SERIAL_ECHOLN(MSG_SD_CANT_OPEN_SUBDIR);
  67. SERIAL_ECHOLN(lfilename);
  68. }
  69. }
  70. lsDive(path,dir);
  71. //close done automatically by destructor of SdFile
  72. }
  73. else
  74. {
  75. if (p.name[0] == DIR_NAME_FREE) break;
  76. if (p.name[0] == DIR_NAME_DELETED || p.name[0] == '.'|| p.name[0] == '_') continue;
  77. if ( p.name[0] == '.')
  78. {
  79. if ( p.name[1] != '.')
  80. continue;
  81. }
  82. if (!DIR_IS_FILE_OR_SUBDIR(&p)) continue;
  83. filenameIsDir=DIR_IS_SUBDIR(&p);
  84. if(!filenameIsDir)
  85. {
  86. if(p.name[8]!='G') continue;
  87. if(p.name[9]=='~') continue;
  88. }
  89. //if(cnt++!=nr) continue;
  90. createFilename(filename,p);
  91. if(lsAction==LS_SerialPrint)
  92. {
  93. SERIAL_PROTOCOL(prepend);
  94. SERIAL_PROTOCOLLN(filename);
  95. }
  96. else if(lsAction==LS_Count)
  97. {
  98. nrFiles++;
  99. }
  100. else if(lsAction==LS_GetFilename)
  101. {
  102. if(cnt==nrFiles)
  103. return;
  104. cnt++;
  105. }
  106. }
  107. }
  108. }
  109. void CardReader::ls()
  110. {
  111. lsAction=LS_SerialPrint;
  112. if(lsAction==LS_Count)
  113. nrFiles=0;
  114. root.rewind();
  115. lsDive("",root);
  116. }
  117. void CardReader::initsd()
  118. {
  119. cardOK = false;
  120. if(root.isOpen())
  121. root.close();
  122. if (!card.init(SPI_FULL_SPEED,SDSS))
  123. {
  124. //if (!card.init(SPI_HALF_SPEED,SDSS))
  125. SERIAL_ECHO_START;
  126. SERIAL_ECHOLNPGM(MSG_SD_INIT_FAIL);
  127. }
  128. else if (!volume.init(&card))
  129. {
  130. SERIAL_ERROR_START;
  131. SERIAL_ERRORLNPGM(MSG_SD_VOL_INIT_FAIL);
  132. }
  133. else if (!root.openRoot(&volume))
  134. {
  135. SERIAL_ERROR_START;
  136. SERIAL_ERRORLNPGM(MSG_SD_OPENROOT_FAIL);
  137. }
  138. else
  139. {
  140. cardOK = true;
  141. SERIAL_ECHO_START;
  142. SERIAL_ECHOLNPGM(MSG_SD_CARD_OK);
  143. }
  144. curDir=&root;
  145. if(!workDir.openRoot(&volume))
  146. {
  147. SERIAL_ECHOLNPGM(MSG_SD_WORKDIR_FAIL);
  148. }
  149. }
  150. void CardReader::setroot()
  151. {
  152. curDir=&root;
  153. if(!workDir.openRoot(&volume))
  154. {
  155. SERIAL_ECHOLNPGM(MSG_SD_WORKDIR_FAIL);
  156. }
  157. }
  158. void CardReader::release()
  159. {
  160. sdprinting = false;
  161. cardOK = false;
  162. }
  163. void CardReader::startFileprint()
  164. {
  165. if(cardOK)
  166. {
  167. sdprinting = true;
  168. }
  169. }
  170. void CardReader::pauseSDPrint()
  171. {
  172. if(sdprinting)
  173. {
  174. sdprinting = false;
  175. }
  176. }
  177. void CardReader::openFile(char* name,bool read)
  178. {
  179. if(!cardOK)
  180. return;
  181. file.close();
  182. sdprinting = false;
  183. SdFile myDir;
  184. curDir=&root;
  185. char *fname=name;
  186. char *dirname_start,*dirname_end;
  187. if(name[0]=='/')
  188. {
  189. dirname_start=strchr(name,'/')+1;
  190. while(dirname_start>0)
  191. {
  192. dirname_end=strchr(dirname_start,'/');
  193. //SERIAL_ECHO("start:");SERIAL_ECHOLN((int)(dirname_start-name));
  194. //SERIAL_ECHO("end :");SERIAL_ECHOLN((int)(dirname_end-name));
  195. if(dirname_end>0 && dirname_end>dirname_start)
  196. {
  197. char subdirname[13];
  198. strncpy(subdirname, dirname_start, dirname_end-dirname_start);
  199. subdirname[dirname_end-dirname_start]=0;
  200. SERIAL_ECHOLN(subdirname);
  201. if(!myDir.open(curDir,subdirname,O_READ))
  202. {
  203. SERIAL_PROTOCOLPGM(MSG_SD_OPEN_FILE_FAIL);
  204. SERIAL_PROTOCOL(subdirname);
  205. SERIAL_PROTOCOLLNPGM(".");
  206. return;
  207. }
  208. else
  209. ;//SERIAL_ECHOLN("dive ok");
  210. curDir=&myDir;
  211. dirname_start=dirname_end+1;
  212. }
  213. else // the reminder after all /fsa/fdsa/ is the filename
  214. {
  215. fname=dirname_start;
  216. //SERIAL_ECHOLN("remaider");
  217. //SERIAL_ECHOLN(fname);
  218. break;
  219. }
  220. }
  221. }
  222. else //relative path
  223. {
  224. curDir=&workDir;
  225. }
  226. if(read)
  227. {
  228. if (file.open(curDir, fname, O_READ))
  229. {
  230. filesize = file.fileSize();
  231. SERIAL_PROTOCOLPGM(MSG_SD_FILE_OPENED);
  232. SERIAL_PROTOCOL(fname);
  233. SERIAL_PROTOCOLPGM(MSG_SD_SIZE);
  234. SERIAL_PROTOCOLLN(filesize);
  235. sdpos = 0;
  236. SERIAL_PROTOCOLLNPGM(MSG_SD_FILE_SELECTED);
  237. LCD_MESSAGE(fname);
  238. }
  239. else
  240. {
  241. SERIAL_PROTOCOLPGM(MSG_SD_OPEN_FILE_FAIL);
  242. SERIAL_PROTOCOL(fname);
  243. SERIAL_PROTOCOLLNPGM(".");
  244. }
  245. }
  246. else
  247. { //write
  248. if (!file.open(curDir, fname, O_CREAT | O_APPEND | O_WRITE | O_TRUNC))
  249. {
  250. SERIAL_PROTOCOLPGM(MSG_SD_OPEN_FILE_FAIL);
  251. SERIAL_PROTOCOL(fname);
  252. SERIAL_PROTOCOLLNPGM(".");
  253. }
  254. else
  255. {
  256. saving = true;
  257. SERIAL_PROTOCOLPGM(MSG_SD_WRITE_TO_FILE);
  258. SERIAL_PROTOCOLLN(name);
  259. LCD_MESSAGE(fname);
  260. }
  261. }
  262. }
  263. void CardReader::removeFile(char* name)
  264. {
  265. if(!cardOK)
  266. return;
  267. file.close();
  268. sdprinting = false;
  269. SdFile myDir;
  270. curDir=&root;
  271. char *fname=name;
  272. char *dirname_start,*dirname_end;
  273. if(name[0]=='/')
  274. {
  275. dirname_start=strchr(name,'/')+1;
  276. while(dirname_start>0)
  277. {
  278. dirname_end=strchr(dirname_start,'/');
  279. //SERIAL_ECHO("start:");SERIAL_ECHOLN((int)(dirname_start-name));
  280. //SERIAL_ECHO("end :");SERIAL_ECHOLN((int)(dirname_end-name));
  281. if(dirname_end>0 && dirname_end>dirname_start)
  282. {
  283. char subdirname[13];
  284. strncpy(subdirname, dirname_start, dirname_end-dirname_start);
  285. subdirname[dirname_end-dirname_start]=0;
  286. SERIAL_ECHOLN(subdirname);
  287. if(!myDir.open(curDir,subdirname,O_READ))
  288. {
  289. SERIAL_PROTOCOLPGM("open failed, File: ");
  290. SERIAL_PROTOCOL(subdirname);
  291. SERIAL_PROTOCOLLNPGM(".");
  292. return;
  293. }
  294. else
  295. ;//SERIAL_ECHOLN("dive ok");
  296. curDir=&myDir;
  297. dirname_start=dirname_end+1;
  298. }
  299. else // the reminder after all /fsa/fdsa/ is the filename
  300. {
  301. fname=dirname_start;
  302. //SERIAL_ECHOLN("remaider");
  303. //SERIAL_ECHOLN(fname);
  304. break;
  305. }
  306. }
  307. }
  308. else //relative path
  309. {
  310. curDir=&workDir;
  311. }
  312. if (file.remove(curDir, fname))
  313. {
  314. SERIAL_PROTOCOLPGM("File deleted:");
  315. SERIAL_PROTOCOL(fname);
  316. sdpos = 0;
  317. }
  318. else
  319. {
  320. SERIAL_PROTOCOLPGM("Deletion failed, File: ");
  321. SERIAL_PROTOCOL(fname);
  322. SERIAL_PROTOCOLLNPGM(".");
  323. }
  324. }
  325. void CardReader::getStatus()
  326. {
  327. if(cardOK){
  328. SERIAL_PROTOCOLPGM(MSG_SD_PRINTING_BYTE);
  329. SERIAL_PROTOCOL(sdpos);
  330. SERIAL_PROTOCOLPGM("/");
  331. SERIAL_PROTOCOLLN(filesize);
  332. }
  333. else{
  334. SERIAL_PROTOCOLLNPGM(MSG_SD_NOT_PRINTING);
  335. }
  336. }
  337. void CardReader::write_command(char *buf)
  338. {
  339. char* begin = buf;
  340. char* npos = 0;
  341. char* end = buf + strlen(buf) - 1;
  342. file.writeError = false;
  343. if((npos = strchr(buf, 'N')) != NULL)
  344. {
  345. begin = strchr(npos, ' ') + 1;
  346. end = strchr(npos, '*') - 1;
  347. }
  348. end[1] = '\r';
  349. end[2] = '\n';
  350. end[3] = '\0';
  351. file.write(begin);
  352. if (file.writeError)
  353. {
  354. SERIAL_ERROR_START;
  355. SERIAL_ERRORLNPGM(MSG_SD_ERR_WRITE_TO_FILE);
  356. }
  357. }
  358. void CardReader::checkautostart(bool force)
  359. {
  360. if(!force)
  361. {
  362. if(!autostart_stilltocheck)
  363. return;
  364. if(autostart_atmillis<millis())
  365. return;
  366. }
  367. autostart_stilltocheck=false;
  368. if(!cardOK)
  369. {
  370. initsd();
  371. if(!cardOK) //fail
  372. return;
  373. }
  374. char autoname[30];
  375. sprintf(autoname,"auto%i.g",lastnr);
  376. for(int8_t i=0;i<(int)strlen(autoname);i++)
  377. autoname[i]=tolower(autoname[i]);
  378. dir_t p;
  379. root.rewind();
  380. bool found=false;
  381. while (root.readDir(p) > 0)
  382. {
  383. for(int8_t i=0;i<(int)strlen((char*)p.name);i++)
  384. p.name[i]=tolower(p.name[i]);
  385. //Serial.print((char*)p.name);
  386. //Serial.print(" ");
  387. //Serial.println(autoname);
  388. if(p.name[9]!='~') //skip safety copies
  389. if(strncmp((char*)p.name,autoname,5)==0)
  390. {
  391. char cmd[30];
  392. sprintf(cmd,"M23 %s",autoname);
  393. enquecommand(cmd);
  394. enquecommand("M24");
  395. found=true;
  396. }
  397. }
  398. if(!found)
  399. lastnr=-1;
  400. else
  401. lastnr++;
  402. }
  403. void CardReader::closefile()
  404. {
  405. file.sync();
  406. file.close();
  407. saving = false;
  408. }
  409. void CardReader::getfilename(const uint8_t nr)
  410. {
  411. curDir=&workDir;
  412. lsAction=LS_GetFilename;
  413. nrFiles=nr;
  414. curDir->rewind();
  415. lsDive("",*curDir);
  416. }
  417. uint16_t CardReader::getnrfilenames()
  418. {
  419. curDir=&workDir;
  420. lsAction=LS_Count;
  421. nrFiles=0;
  422. curDir->rewind();
  423. lsDive("",*curDir);
  424. //SERIAL_ECHOLN(nrFiles);
  425. return nrFiles;
  426. }
  427. void CardReader::chdir(const char * relpath)
  428. {
  429. SdFile newfile;
  430. SdFile *parent=&root;
  431. if(workDir.isOpen())
  432. parent=&workDir;
  433. if(!newfile.open(*parent,relpath, O_READ))
  434. {
  435. SERIAL_ECHO_START;
  436. SERIAL_ECHOPGM(MSG_SD_CANT_ENTER_SUBDIR);
  437. SERIAL_ECHOLN(relpath);
  438. }
  439. else
  440. {
  441. workDirParentParent=workDirParent;
  442. workDirParent=*parent;
  443. workDir=newfile;
  444. }
  445. }
  446. void CardReader::updir()
  447. {
  448. if(!workDir.isRoot())
  449. {
  450. workDir=workDirParent;
  451. workDirParent=workDirParentParent;
  452. }
  453. }
  454. void CardReader::printingHasFinished()
  455. {
  456. st_synchronize();
  457. quickStop();
  458. sdprinting = false;
  459. if(SD_FINISHED_STEPPERRELEASE)
  460. {
  461. //finishAndDisableSteppers();
  462. enquecommand(SD_FINISHED_RELEASECOMMAND);
  463. }
  464. autotempShutdown();
  465. }
  466. #endif //SDSUPPORT