123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688 |
- #include "cmdqueue.h"
- #include "cardreader.h"
- #include "ultralcd.h"
- extern bool Stopped;
- char cmdbuffer[BUFSIZE * (MAX_CMD_SIZE + 1) + CMDBUFFER_RESERVE_FRONT];
- int bufindr = 0;
- int bufindw = 0;
- int buflen = 0;
- bool cmdbuffer_front_already_processed = false;
- int serial_count = 0;
- boolean comment_mode = false;
- char *strchr_pointer;
- unsigned long TimeSent = millis();
- unsigned long TimeNow = millis();
- long gcode_N = 0;
- long gcode_LastN = 0;
- long Stopped_gcode_LastN = 0;
- uint32_t sdpos_atomic = 0;
- bool cmdqueue_pop_front()
- {
- if (buflen > 0) {
- #ifdef CMDBUFFER_DEBUG
- SERIAL_ECHOPGM("Dequeing ");
- SERIAL_ECHO(cmdbuffer+bufindr+CMDHDRSIZE);
- SERIAL_ECHOLNPGM("");
- SERIAL_ECHOPGM("Old indices: buflen ");
- SERIAL_ECHO(buflen);
- SERIAL_ECHOPGM(", bufindr ");
- SERIAL_ECHO(bufindr);
- SERIAL_ECHOPGM(", bufindw ");
- SERIAL_ECHO(bufindw);
- SERIAL_ECHOPGM(", serial_count ");
- SERIAL_ECHO(serial_count);
- SERIAL_ECHOPGM(", bufsize ");
- SERIAL_ECHO(sizeof(cmdbuffer));
- SERIAL_ECHOLNPGM("");
- #endif
- if (-- buflen == 0) {
-
- if (serial_count == 0)
-
- bufindw = 0;
- bufindr = bufindw;
- } else {
-
-
- for (bufindr += CMDHDRSIZE; cmdbuffer[bufindr] != 0; ++ bufindr) ;
-
- for (++ bufindr; bufindr < sizeof(cmdbuffer) && cmdbuffer[bufindr] == 0; ++ bufindr) ;
-
- if (bufindr == sizeof(cmdbuffer)) {
-
- for (bufindr = 0; cmdbuffer[bufindr] == 0; ++ bufindr) ;
- }
- #ifdef CMDBUFFER_DEBUG
- SERIAL_ECHOPGM("New indices: buflen ");
- SERIAL_ECHO(buflen);
- SERIAL_ECHOPGM(", bufindr ");
- SERIAL_ECHO(bufindr);
- SERIAL_ECHOPGM(", bufindw ");
- SERIAL_ECHO(bufindw);
- SERIAL_ECHOPGM(", serial_count ");
- SERIAL_ECHO(serial_count);
- SERIAL_ECHOPGM(" new command on the top: ");
- SERIAL_ECHO(cmdbuffer+bufindr+CMDHDRSIZE);
- SERIAL_ECHOLNPGM("");
- #endif
- }
- return true;
- }
- return false;
- }
- void cmdqueue_reset()
- {
- bufindr = 0;
- bufindw = 0;
- buflen = 0;
- cmdbuffer_front_already_processed = false;
- }
- bool cmdqueue_could_enqueue_front(int len_asked)
- {
-
- if (len_asked >= MAX_CMD_SIZE)
- return false;
-
- if (! cmdbuffer_front_already_processed) {
- cmdqueue_pop_front();
- cmdbuffer_front_already_processed = true;
- }
- if (bufindr == bufindw && buflen > 0)
-
- return false;
-
- int endw = (serial_count > 0) ? (bufindw + MAX_CMD_SIZE + 1) : bufindw;
- if (bufindw < bufindr) {
- int bufindr_new = bufindr - len_asked - (1 + CMDHDRSIZE);
-
- if (endw <= bufindr_new) {
- bufindr = bufindr_new;
- return true;
- }
- } else {
-
- if (len_asked + (1 + CMDHDRSIZE) <= bufindr) {
-
- bufindr -= len_asked + (1 + CMDHDRSIZE);
- return true;
- }
- int bufindr_new = sizeof(cmdbuffer) - len_asked - (1 + CMDHDRSIZE);
- if (endw <= bufindr_new) {
- memset(cmdbuffer, 0, bufindr);
- bufindr = bufindr_new;
- return true;
- }
- }
- return false;
- }
- bool cmdqueue_could_enqueue_back(int len_asked, bool atomic_update)
- {
-
- if (len_asked >= MAX_CMD_SIZE)
- return false;
- if (bufindr == bufindw && buflen > 0)
-
- return false;
- if (serial_count > 0) {
-
-
-
-
-
- int endw = bufindw + len_asked + (1 + CMDHDRSIZE);
- if (bufindw < bufindr)
-
- return endw + CMDBUFFER_RESERVE_FRONT <= bufindr;
-
- if (
- endw + CMDBUFFER_RESERVE_FRONT <= sizeof(cmdbuffer) ||
-
- (endw <= sizeof(cmdbuffer) && CMDBUFFER_RESERVE_FRONT <= bufindr))
- return true;
-
- if (len_asked + (1 + CMDHDRSIZE) + CMDBUFFER_RESERVE_FRONT <= bufindr) {
-
- memset(cmdbuffer+bufindw, 0, sizeof(cmdbuffer)-bufindw);
-
-
- if (atomic_update)
- cli();
- bufindw = 0;
- if (atomic_update)
- sei();
- return true;
- }
- } else {
-
-
- int endw = bufindw + len_asked + (1 + CMDHDRSIZE);
- if (bufindw < bufindr)
-
- return endw + CMDBUFFER_RESERVE_FRONT <= bufindr;
-
- if (
- endw + CMDBUFFER_RESERVE_FRONT <= sizeof(cmdbuffer) ||
-
- (endw <= sizeof(cmdbuffer) && CMDBUFFER_RESERVE_FRONT <= bufindr))
- return true;
-
- if (len_asked + (1 + CMDHDRSIZE) + CMDBUFFER_RESERVE_FRONT <= bufindr) {
-
- memset(cmdbuffer+bufindw, 0, sizeof(cmdbuffer)-bufindw);
-
-
- if (atomic_update)
- cli();
- bufindw = 0;
- if (atomic_update)
- sei();
- return true;
- }
- }
- return false;
- }
- #ifdef CMDBUFFER_DEBUG
- void cmdqueue_dump_to_serial_single_line(int nr, const char *p)
- {
- SERIAL_ECHOPGM("Entry nr: ");
- SERIAL_ECHO(nr);
- SERIAL_ECHOPGM(", type: ");
- SERIAL_ECHO(int(*p));
- SERIAL_ECHOPGM(", cmd: ");
- SERIAL_ECHO(p+1);
- SERIAL_ECHOLNPGM("");
- }
- void cmdqueue_dump_to_serial()
- {
- if (buflen == 0) {
- SERIAL_ECHOLNPGM("The command buffer is empty.");
- } else {
- SERIAL_ECHOPGM("Content of the buffer: entries ");
- SERIAL_ECHO(buflen);
- SERIAL_ECHOPGM(", indr ");
- SERIAL_ECHO(bufindr);
- SERIAL_ECHOPGM(", indw ");
- SERIAL_ECHO(bufindw);
- SERIAL_ECHOLNPGM("");
- int nr = 0;
- if (bufindr < bufindw) {
- for (const char *p = cmdbuffer + bufindr; p < cmdbuffer + bufindw; ++ nr) {
- cmdqueue_dump_to_serial_single_line(nr, p);
-
- for (++p; *p != 0; ++ p);
-
- for (++p; p < cmdbuffer + bufindw && *p == 0; ++ p);
- }
- } else {
- for (const char *p = cmdbuffer + bufindr; p < cmdbuffer + sizeof(cmdbuffer); ++ nr) {
- cmdqueue_dump_to_serial_single_line(nr, p);
-
- for (++p; *p != 0; ++ p);
-
- for (++p; p < cmdbuffer + sizeof(cmdbuffer) && *p == 0; ++ p);
- }
- for (const char *p = cmdbuffer; p < cmdbuffer + bufindw; ++ nr) {
- cmdqueue_dump_to_serial_single_line(nr, p);
-
- for (++p; *p != 0; ++ p);
-
- for (++p; p < cmdbuffer + bufindw && *p == 0; ++ p);
- }
- }
- SERIAL_ECHOLNPGM("End of the buffer.");
- }
- }
- #endif
- void enquecommand(const char *cmd, bool from_progmem)
- {
- int len = from_progmem ? strlen_P(cmd) : strlen(cmd);
-
-
- if (cmdqueue_could_enqueue_back(len)) {
-
-
- cmdbuffer[bufindw] = CMDBUFFER_CURRENT_TYPE_UI;
- if (from_progmem)
- strcpy_P(cmdbuffer + bufindw + CMDHDRSIZE, cmd);
- else
- strcpy(cmdbuffer + bufindw + CMDHDRSIZE, cmd);
- SERIAL_ECHO_START;
- SERIAL_ECHORPGM(MSG_Enqueing);
- SERIAL_ECHO(cmdbuffer + bufindw + CMDHDRSIZE);
- SERIAL_ECHOLNPGM("\"");
- bufindw += len + (CMDHDRSIZE + 1);
- if (bufindw == sizeof(cmdbuffer))
- bufindw = 0;
- ++ buflen;
- #ifdef CMDBUFFER_DEBUG
- cmdqueue_dump_to_serial();
- #endif
- } else {
- SERIAL_ERROR_START;
- SERIAL_ECHORPGM(MSG_Enqueing);
- if (from_progmem)
- SERIAL_PROTOCOLRPGM(cmd);
- else
- SERIAL_ECHO(cmd);
- SERIAL_ECHOLNPGM("\" failed: Buffer full!");
- #ifdef CMDBUFFER_DEBUG
- cmdqueue_dump_to_serial();
- #endif
- }
- }
- bool cmd_buffer_empty()
- {
- return (buflen == 0);
- }
- void enquecommand_front(const char *cmd, bool from_progmem)
- {
- int len = from_progmem ? strlen_P(cmd) : strlen(cmd);
-
- if (cmdqueue_could_enqueue_front(len)) {
- cmdbuffer[bufindr] = CMDBUFFER_CURRENT_TYPE_UI;
- if (from_progmem)
- strcpy_P(cmdbuffer + bufindr + CMDHDRSIZE, cmd);
- else
- strcpy(cmdbuffer + bufindr + CMDHDRSIZE, cmd);
- ++ buflen;
- SERIAL_ECHO_START;
- SERIAL_ECHOPGM("Enqueing to the front: \"");
- SERIAL_ECHO(cmdbuffer + bufindr + CMDHDRSIZE);
- SERIAL_ECHOLNPGM("\"");
- #ifdef CMDBUFFER_DEBUG
- cmdqueue_dump_to_serial();
- #endif
- } else {
- SERIAL_ERROR_START;
- SERIAL_ECHOPGM("Enqueing to the front: \"");
- if (from_progmem)
- SERIAL_PROTOCOLRPGM(cmd);
- else
- SERIAL_ECHO(cmd);
- SERIAL_ECHOLNPGM("\" failed: Buffer full!");
- #ifdef CMDBUFFER_DEBUG
- cmdqueue_dump_to_serial();
- #endif
- }
- }
- void repeatcommand_front()
- {
- cmdbuffer_front_already_processed = true;
- }
- bool is_buffer_empty()
- {
- if (buflen == 0) return true;
- else return false;
- }
- void get_command()
- {
-
- if (! cmdqueue_could_enqueue_back(MAX_CMD_SIZE - 1, true))
- return;
-
- bool rx_buffer_full = false;
- if (MYSERIAL.available() == RX_BUFFER_SIZE - 1) {
- MYSERIAL.flush();
- SERIAL_ECHOLNPGM("Full RX Buffer");
- rx_buffer_full = true;
- }
- while (MYSERIAL.available() > 0) {
- char serial_char = MYSERIAL.read();
-
- TimeSent = millis();
- TimeNow = millis();
- if (serial_char < 0)
-
-
-
-
- continue;
- if(serial_char == '\n' ||
- serial_char == '\r' ||
- serial_count >= (MAX_CMD_SIZE - 1) )
- {
- if(!serial_count) {
- comment_mode = false;
- return;
- }
- cmdbuffer[bufindw+serial_count+CMDHDRSIZE] = 0;
- if(!comment_mode){
-
- if ((strstr(cmdbuffer+bufindw+CMDHDRSIZE, "PRUSA") == NULL) &&
- (cmdbuffer[bufindw+CMDHDRSIZE] == 'N')) {
-
-
- gcode_N = (strtol(cmdbuffer+bufindw+CMDHDRSIZE+1, NULL, 10));
- if(gcode_N != gcode_LastN+1 && (strstr_P(cmdbuffer+bufindw+CMDHDRSIZE, PSTR("M110")) == NULL) ) {
-
-
- SERIAL_ERROR_START;
- SERIAL_ERRORRPGM(MSG_ERR_LINE_NO);
- SERIAL_ERRORLN(gcode_LastN);
-
- FlushSerialRequestResend();
- serial_count = 0;
- return;
- }
- if((strchr_pointer = strchr(cmdbuffer+bufindw+CMDHDRSIZE, '*')) != NULL)
- {
- byte checksum = 0;
- char *p = cmdbuffer+bufindw+CMDHDRSIZE;
- while (p != strchr_pointer)
- checksum = checksum^(*p++);
- if (int(strtol(strchr_pointer+1, NULL, 10)) != int(checksum)) {
- SERIAL_ERROR_START;
- SERIAL_ERRORRPGM(MSG_ERR_CHECKSUM_MISMATCH);
- SERIAL_ERRORLN(gcode_LastN);
- FlushSerialRequestResend();
- serial_count = 0;
- return;
- }
-
- *strchr_pointer = 0;
- }
- else
- {
- SERIAL_ERROR_START;
- SERIAL_ERRORRPGM(MSG_ERR_NO_CHECKSUM);
- SERIAL_ERRORLN(gcode_LastN);
- FlushSerialRequestResend();
- serial_count = 0;
- return;
- }
-
- cmdbuffer[bufindw + CMDHDRSIZE] = '$';
-
- gcode_LastN = gcode_N;
- }
-
- if ((cmdbuffer[bufindw + CMDHDRSIZE] != 'N') && (cmdbuffer[bufindw + CMDHDRSIZE] != '$') && (strchr(cmdbuffer+bufindw+CMDHDRSIZE, '*') != NULL))
- {
- SERIAL_ERROR_START;
- SERIAL_ERRORRPGM(MSG_ERR_NO_LINENUMBER_WITH_CHECKSUM);
- SERIAL_ERRORLN(gcode_LastN);
- serial_count = 0;
- return;
- }
- if ((strchr_pointer = strchr(cmdbuffer+bufindw+CMDHDRSIZE, 'G')) != NULL) {
- if (! IS_SD_PRINTING) {
- usb_printing_counter = 10;
- is_usb_printing = true;
- }
- if (Stopped == true) {
- int gcode = strtol(strchr_pointer+1, NULL, 10);
- if (gcode >= 0 && gcode <= 3) {
- SERIAL_ERRORLNRPGM(MSG_ERR_STOPPED);
- LCD_MESSAGERPGM(MSG_STOPPED);
- }
- }
- }
-
- if(strcmp(cmdbuffer+bufindw+CMDHDRSIZE, "M112") == 0)
- kill("", 2);
-
-
- cmdbuffer[bufindw] = CMDBUFFER_CURRENT_TYPE_USB;
- #ifdef CMDBUFFER_DEBUG
- SERIAL_ECHO_START;
- SERIAL_ECHOPGM("Storing a command line to buffer: ");
- SERIAL_ECHO(cmdbuffer+bufindw+CMDHDRSIZE);
- SERIAL_ECHOLNPGM("");
- #endif
- bufindw += strlen(cmdbuffer+bufindw+CMDHDRSIZE) + (1 + CMDHDRSIZE);
- if (bufindw == sizeof(cmdbuffer))
- bufindw = 0;
- ++ buflen;
- #ifdef CMDBUFFER_DEBUG
- SERIAL_ECHOPGM("Number of commands in the buffer: ");
- SERIAL_ECHO(buflen);
- SERIAL_ECHOLNPGM("");
- #endif
- }
- serial_count = 0;
-
-
- if (MYSERIAL.available() == 0 || ! cmdqueue_could_enqueue_back(MAX_CMD_SIZE-1, true))
- return;
- }
- else {
-
- if(serial_char == ';') comment_mode = true;
- if(!comment_mode) cmdbuffer[bufindw+CMDHDRSIZE+serial_count++] = serial_char;
- }
- }
- if(farm_mode){
- TimeNow = millis();
- if ( ((TimeNow - TimeSent) > 800) && (serial_count > 0) ) {
- cmdbuffer[bufindw+serial_count+CMDHDRSIZE] = 0;
-
- bufindw += strlen(cmdbuffer+bufindw+CMDHDRSIZE) + (1 + CMDHDRSIZE);
- if (bufindw == sizeof(cmdbuffer))
- bufindw = 0;
- ++ buflen;
-
- serial_count = 0;
-
- SERIAL_ECHOPGM("TIMEOUT:");
-
- return;
- }
- }
-
-
- #ifdef SDSUPPORT
- if(!card.sdprinting || serial_count!=0){
-
-
- return;
- }
-
-
-
- static bool stop_buffering=false;
- if(buflen==0) stop_buffering=false;
- union {
- struct {
- char lo;
- char hi;
- } lohi;
- uint16_t value;
- } sd_count;
- sd_count.value = 0;
-
- while( !card.eof() && !stop_buffering) {
- int16_t n=card.get();
- char serial_char = (char)n;
- if(serial_char == '\n' ||
- serial_char == '\r' ||
- ((serial_char == '#' || serial_char == ':') && comment_mode == false) ||
- serial_count >= (MAX_CMD_SIZE - 1) || n==-1)
- {
- if(card.eof()){
- SERIAL_PROTOCOLLNRPGM(MSG_FILE_PRINTED);
- stoptime=millis();
- char time[30];
- unsigned long t=(stoptime-starttime-pause_time)/1000;
- pause_time = 0;
- int hours, minutes;
- minutes=(t/60)%60;
- hours=t/60/60;
- save_statistics(total_filament_used, t);
- sprintf_P(time, PSTR("%i hours %i minutes"),hours, minutes);
- SERIAL_ECHO_START;
- SERIAL_ECHOLN(time);
- lcd_setstatus(time);
- card.printingHasFinished();
- card.checkautostart(true);
- if (farm_mode)
- {
- prusa_statistics(6);
- lcd_commands_type = LCD_COMMAND_FARM_MODE_CONFIRM;
- }
- }
- if(serial_char=='#')
- stop_buffering=true;
- if(!serial_count)
- {
-
-
-
-
-
- comment_mode = false;
- continue;
- }
-
-
- sd_count.value = (card.get_sdpos()+1) - sdpos_atomic;
- cmdbuffer[bufindw] = CMDBUFFER_CURRENT_TYPE_SDCARD;
- cmdbuffer[bufindw+1] = sd_count.lohi.lo;
- cmdbuffer[bufindw+2] = sd_count.lohi.hi;
- cmdbuffer[bufindw+serial_count+CMDHDRSIZE] = 0;
-
- uint8_t len = strlen(cmdbuffer+bufindw+CMDHDRSIZE) + (1 + CMDHDRSIZE);
- sd_count.value = 0;
- cli();
-
-
-
-
- ++ buflen;
- bufindw += len;
- sdpos_atomic = card.get_sdpos()+1;
- if (bufindw == sizeof(cmdbuffer))
- bufindw = 0;
- sei();
- comment_mode = false;
- serial_count = 0;
-
- if (! cmdqueue_could_enqueue_back(MAX_CMD_SIZE-1, true))
- return;
- }
- else
- {
- if(serial_char == ';') comment_mode = true;
- else if(!comment_mode) cmdbuffer[bufindw+CMDHDRSIZE+serial_count++] = serial_char;
- }
- }
- #endif
- }
- uint16_t cmdqueue_calc_sd_length()
- {
- if (buflen == 0)
- return 0;
- union {
- struct {
- char lo;
- char hi;
- } lohi;
- uint16_t value;
- } sdlen_single;
- uint16_t sdlen = 0;
- for (int _buflen = buflen, _bufindr = bufindr;;) {
- if (cmdbuffer[_bufindr] == CMDBUFFER_CURRENT_TYPE_SDCARD) {
- sdlen_single.lohi.lo = cmdbuffer[_bufindr + 1];
- sdlen_single.lohi.hi = cmdbuffer[_bufindr + 2];
- sdlen += sdlen_single.value;
- }
- if (-- _buflen == 0)
- break;
-
- for (_bufindr += CMDHDRSIZE; cmdbuffer[_bufindr] != 0; ++ _bufindr) ;
-
- for (++ _bufindr; _bufindr < sizeof(cmdbuffer) && cmdbuffer[_bufindr] == 0; ++ _bufindr) ;
-
- if (_bufindr == sizeof(cmdbuffer)) {
-
- for (_bufindr = 0; cmdbuffer[_bufindr] == 0; ++ _bufindr) ;
- }
- }
- return sdlen;
- }
|