2012-11-01 16:37:27 +01:00
|
|
|
/*
|
|
|
|
report.c - reporting and messaging methods
|
|
|
|
Part of Grbl
|
|
|
|
|
|
|
|
Copyright (c) 2012 Sungeun K. Jeon
|
|
|
|
|
|
|
|
Grbl 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.
|
|
|
|
|
|
|
|
Grbl 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 Grbl. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
This file functions as the primary feedback interface for Grbl. Any outgoing data, such
|
|
|
|
as the protocol status messages, feedback messages, and status reports, are stored here.
|
|
|
|
For the most part, these functions primarily are called from protocol.c methods. If a
|
|
|
|
different style feedback is desired (i.e. JSON), then a user can change these following
|
|
|
|
methods to accomodate their needs.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <avr/io.h>
|
|
|
|
#include "print.h"
|
|
|
|
#include "settings.h"
|
|
|
|
#include <math.h>
|
|
|
|
#include "nuts_bolts.h"
|
|
|
|
#include <avr/pgmspace.h>
|
|
|
|
#include "report.h"
|
|
|
|
#include "protocol.h"
|
2012-11-02 02:48:55 +01:00
|
|
|
#include "gcode.h"
|
|
|
|
#include "coolant_control.h"
|
2012-11-01 16:37:27 +01:00
|
|
|
|
|
|
|
|
|
|
|
// Handles the primary confirmation protocol response for streaming interfaces and human-feedback.
|
|
|
|
// For every incoming line, this method responds with an 'ok' for a successful command or an
|
|
|
|
// 'error:' to indicate some error event with the line or some critical system error during
|
|
|
|
// operation. Errors events can originate from the g-code parser, settings module, or asynchronously
|
|
|
|
// from a critical error, such as a triggered hard limit. Interface should always monitor for these
|
|
|
|
// responses.
|
|
|
|
// NOTE: In silent mode, all error codes are greater than zero.
|
|
|
|
// TODO: Install silent mode to return only numeric values, primarily for GUIs.
|
|
|
|
void report_status_message(uint8_t status_code)
|
|
|
|
{
|
|
|
|
if (status_code == 0) { // STATUS_OK
|
|
|
|
printPgmString(PSTR("ok\r\n"));
|
|
|
|
} else {
|
|
|
|
printPgmString(PSTR("error: "));
|
|
|
|
switch(status_code) {
|
|
|
|
case STATUS_BAD_NUMBER_FORMAT:
|
|
|
|
printPgmString(PSTR("Bad number format")); break;
|
|
|
|
case STATUS_EXPECTED_COMMAND_LETTER:
|
|
|
|
printPgmString(PSTR("Expected command letter")); break;
|
|
|
|
case STATUS_UNSUPPORTED_STATEMENT:
|
|
|
|
printPgmString(PSTR("Unsupported statement")); break;
|
|
|
|
case STATUS_FLOATING_POINT_ERROR:
|
|
|
|
printPgmString(PSTR("Floating point error")); break;
|
|
|
|
case STATUS_MODAL_GROUP_VIOLATION:
|
|
|
|
printPgmString(PSTR("Modal group violation")); break;
|
|
|
|
case STATUS_INVALID_STATEMENT:
|
|
|
|
printPgmString(PSTR("Invalid statement")); break;
|
|
|
|
case STATUS_HARD_LIMIT:
|
|
|
|
printPgmString(PSTR("Limit triggered")); break;
|
|
|
|
case STATUS_SETTING_DISABLED:
|
|
|
|
printPgmString(PSTR("Grbl setting disabled")); break;
|
|
|
|
case STATUS_SETTING_VALUE_NEG:
|
|
|
|
printPgmString(PSTR("Set value must be > 0.0")); break;
|
|
|
|
case STATUS_SETTING_STEP_PULSE_MIN:
|
|
|
|
printPgmString(PSTR("Step pulse must be >= 3 microseconds")); break;
|
|
|
|
case STATUS_SETTING_READ_FAIL:
|
|
|
|
printPgmString(PSTR("Failed to read EEPROM settings. Using defaults")); break;
|
|
|
|
}
|
|
|
|
printPgmString(PSTR("\r\n"));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Prints feedback messages. This serves as a centralized method to provide additional
|
|
|
|
// user feedback for things that are not of the status message response protocol. These
|
|
|
|
// are messages such as setup warnings and how to exit alarms.
|
2012-11-02 02:48:55 +01:00
|
|
|
// NOTE: For interfaces, messages are always placed within chevrons. And if silent mode
|
|
|
|
// is installed, the message number codes are less than zero.
|
2012-11-01 16:37:27 +01:00
|
|
|
// TODO: Install silence feedback messages option in settings
|
|
|
|
void report_feedback_message(int8_t message_code)
|
|
|
|
{
|
2012-11-02 02:48:55 +01:00
|
|
|
printPgmString(PSTR("<"));
|
2012-11-01 16:37:27 +01:00
|
|
|
switch(message_code) {
|
|
|
|
case MESSAGE_SYSTEM_ALARM:
|
|
|
|
printPgmString(PSTR("ALARM: Check and reset Grbl")); break;
|
|
|
|
case MESSAGE_POSITION_LOST:
|
|
|
|
printPgmString(PSTR("warning: Position may be lost")); break;
|
|
|
|
case MESSAGE_HOMING_ENABLE:
|
|
|
|
printPgmString(PSTR("'$H' to home. Ensure all limit switches are installed")); break;
|
2012-11-02 02:48:55 +01:00
|
|
|
case MESSAGE_SWITCH_ON:
|
|
|
|
printPgmString(PSTR("Switch enabled")); break;
|
|
|
|
case MESSAGE_SWITCH_OFF:
|
|
|
|
printPgmString(PSTR("Switch disabled")); break;
|
2012-11-01 16:37:27 +01:00
|
|
|
}
|
2012-11-02 02:48:55 +01:00
|
|
|
printPgmString(PSTR(">\r\n"));
|
2012-11-01 16:37:27 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Welcome message
|
|
|
|
void report_init_message()
|
|
|
|
{
|
2012-11-02 02:48:55 +01:00
|
|
|
printPgmString(PSTR("\r\nGrbl " GRBL_VERSION " ['$' for help]\r\n"));
|
2012-11-01 16:37:27 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void report_grbl_help() {
|
2012-11-02 02:48:55 +01:00
|
|
|
// char st_line[LINE_BUFFER_SIZE];
|
|
|
|
// printPgmString(PSTR("\r\n\r\n Startup\r\n$100 = ")); printString(settings_read_startup(st_line,0));
|
|
|
|
// printPgmString(PSTR("\r\n$101 = ")); printString(settings_read_startup(st_line,1));
|
|
|
|
|
|
|
|
// char buf[4];
|
|
|
|
// settings_startup_string((char *)buf);
|
|
|
|
// printPgmString(PSTR("\r\n Startup: ")); printString(buf);
|
|
|
|
|
|
|
|
printPgmString(PSTR("$ (help)\r\n$$ (print Grbl settings)\r\n$# (print gcode parameters)\r\n$G (print gcode parser state)"));
|
|
|
|
printPgmString(PSTR("\r\n$x=value (store Grbl setting)\r\n$Nx=line (store startup block)\r\n$H (run homing cycle, if enabled)"));
|
|
|
|
printPgmString(PSTR("\r\n$B (toggle block delete)\r\n$S (toggle single block mode)\r\n$O (toggle opt stop)"));
|
|
|
|
printPgmString(PSTR("\r\n~ (cycle start)\r\n! (feed hold)\r\n? (current position)\r\n^x (reset Grbl)\r\n"));
|
|
|
|
}
|
|
|
|
|
|
|
|
void report_grbl_settings() {
|
2012-11-01 16:37:27 +01:00
|
|
|
printPgmString(PSTR("$0 = ")); printFloat(settings.steps_per_mm[X_AXIS]);
|
|
|
|
printPgmString(PSTR(" (x axis, steps/mm)\r\n$1 = ")); printFloat(settings.steps_per_mm[Y_AXIS]);
|
|
|
|
printPgmString(PSTR(" (y axis, steps/mm)\r\n$2 = ")); printFloat(settings.steps_per_mm[Z_AXIS]);
|
|
|
|
printPgmString(PSTR(" (z axis, steps/mm)\r\n$3 = ")); printInteger(settings.pulse_microseconds);
|
|
|
|
printPgmString(PSTR(" (step pulse, usec)\r\n$4 = ")); printFloat(settings.default_feed_rate);
|
|
|
|
printPgmString(PSTR(" (default feed rate, mm/min)\r\n$5 = ")); printFloat(settings.default_seek_rate);
|
|
|
|
printPgmString(PSTR(" (default seek rate, mm/min)\r\n$6 = ")); printFloat(settings.mm_per_arc_segment);
|
|
|
|
printPgmString(PSTR(" (arc resolution, mm/segment)\r\n$7 = ")); printInteger(settings.invert_mask);
|
|
|
|
printPgmString(PSTR(" (step port invert mask, int:binary = ")); print_uint8_base2(settings.invert_mask);
|
|
|
|
printPgmString(PSTR(")\r\n$8 = ")); printFloat(settings.acceleration/(60*60)); // Convert from mm/min^2 for human readability
|
|
|
|
printPgmString(PSTR(" (acceleration, mm/sec^2)\r\n$9 = ")); printFloat(settings.junction_deviation);
|
|
|
|
printPgmString(PSTR(" (cornering junction deviation, mm)\r\n$10 = ")); printInteger(bit_istrue(settings.flags,BITFLAG_REPORT_INCHES));
|
|
|
|
printPgmString(PSTR(" (status report inches, bool)\r\n$11 = ")); printInteger(bit_istrue(settings.flags,BITFLAG_AUTO_START));
|
|
|
|
printPgmString(PSTR(" (auto start enable, bool)\r\n$12 = ")); printInteger(bit_istrue(settings.flags,BITFLAG_INVERT_ST_ENABLE));
|
|
|
|
printPgmString(PSTR(" (invert stepper enable, bool)\r\n$13 = ")); printInteger(bit_istrue(settings.flags,BITFLAG_HARD_LIMIT_ENABLE));
|
|
|
|
printPgmString(PSTR(" (hard limit enable, bool)\r\n$14 = ")); printInteger(bit_istrue(settings.flags,BITFLAG_HOMING_ENABLE));
|
|
|
|
printPgmString(PSTR(" (homing enable, bool)\r\n$15 = ")); printInteger(settings.homing_dir_mask);
|
|
|
|
printPgmString(PSTR(" (homing direction mask, int:binary = ")); print_uint8_base2(settings.homing_dir_mask);
|
|
|
|
printPgmString(PSTR(")\r\n$16 = ")); printFloat(settings.homing_feed_rate);
|
|
|
|
printPgmString(PSTR(" (homing feed rate, mm/min)\r\n$17 = ")); printFloat(settings.homing_seek_rate);
|
|
|
|
printPgmString(PSTR(" (homing seek rate, mm/min)\r\n$18 = ")); printInteger(settings.homing_debounce_delay);
|
|
|
|
printPgmString(PSTR(" (homing debounce delay, msec)\r\n$19 = ")); printFloat(settings.homing_pulloff);
|
|
|
|
printPgmString(PSTR(" (homing pull-off travel, mm)\r\n$20 = ")); printInteger(settings.stepper_idle_lock_time);
|
|
|
|
printPgmString(PSTR(" (stepper idle lock time, msec)\r\n$21 = ")); printInteger(settings.decimal_places);
|
|
|
|
printPgmString(PSTR(" (decimal places, int)\r\n$22 = ")); printInteger(settings.n_arc_correction);
|
2012-11-02 02:48:55 +01:00
|
|
|
printPgmString(PSTR(" (n arc correction, int)\r\n"));
|
2012-11-01 16:37:27 +01:00
|
|
|
}
|
|
|
|
|
2012-11-02 02:48:55 +01:00
|
|
|
void report_gcode_parameters()
|
|
|
|
{
|
|
|
|
float coord_data[N_AXIS];
|
|
|
|
uint8_t coord_select, i;
|
|
|
|
for (coord_select = 0; coord_select <= SETTING_INDEX_NCOORD; coord_select++) {
|
|
|
|
if (!(settings_read_coord_data(coord_select,coord_data))) {
|
|
|
|
report_status_message(STATUS_SETTING_READ_FAIL);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
switch (coord_select) {
|
|
|
|
case 0: printPgmString(PSTR("G54")); break;
|
|
|
|
case 1: printPgmString(PSTR("G55")); break;
|
|
|
|
case 2: printPgmString(PSTR("G56")); break;
|
|
|
|
case 3: printPgmString(PSTR("G57")); break;
|
|
|
|
case 4: printPgmString(PSTR("G58")); break;
|
|
|
|
case 5: printPgmString(PSTR("G59")); break;
|
|
|
|
case 6: printPgmString(PSTR("G28")); break;
|
|
|
|
case 7: printPgmString(PSTR("G30")); break;
|
|
|
|
// case 8: printPgmString(PSTR("G92")); break; // G92.2, G92.3 currently not supported.
|
|
|
|
}
|
|
|
|
printPgmString(PSTR(":["));
|
|
|
|
for (i=0; i<N_AXIS; i++) {
|
|
|
|
if (bit_istrue(settings.flags,BITFLAG_REPORT_INCHES)) { printFloat(coord_data[i]*INCH_PER_MM); }
|
|
|
|
else { printFloat(coord_data[i]); }
|
|
|
|
if (i < (N_AXIS-1)) { printPgmString(PSTR(",")); }
|
|
|
|
else { printPgmString(PSTR("]\r\n")); }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
printPgmString(PSTR("G92:[")); // Print G92,G92.1 which are not persistent in memory
|
|
|
|
for (i=0; i<N_AXIS; i++) {
|
|
|
|
if (bit_istrue(settings.flags,BITFLAG_REPORT_INCHES)) { printFloat(gc.coord_offset[i]*INCH_PER_MM); }
|
|
|
|
else { printFloat(gc.coord_offset[i]); }
|
|
|
|
if (i < (N_AXIS-1)) { printPgmString(PSTR(",")); }
|
|
|
|
else { printPgmString(PSTR("]\r\n")); }
|
|
|
|
}
|
|
|
|
}
|
2012-11-01 16:37:27 +01:00
|
|
|
|
2012-11-02 02:48:55 +01:00
|
|
|
void report_gcode_modes()
|
|
|
|
{
|
|
|
|
switch (gc.motion_mode) {
|
|
|
|
case MOTION_MODE_SEEK : printPgmString(PSTR("G0")); break;
|
|
|
|
case MOTION_MODE_LINEAR : printPgmString(PSTR("G1")); break;
|
|
|
|
case MOTION_MODE_CW_ARC : printPgmString(PSTR("G2")); break;
|
|
|
|
case MOTION_MODE_CCW_ARC : printPgmString(PSTR("G3")); break;
|
|
|
|
case MOTION_MODE_CANCEL : printPgmString(PSTR("G80")); break;
|
|
|
|
}
|
2012-11-01 16:37:27 +01:00
|
|
|
|
2012-11-02 02:48:55 +01:00
|
|
|
printPgmString(PSTR(" G"));
|
|
|
|
printInteger(gc.coord_select+54);
|
|
|
|
|
|
|
|
if (gc.plane_axis_0 == X_AXIS) {
|
|
|
|
if (gc.plane_axis_1 == Y_AXIS) { printPgmString(PSTR(" G17")); }
|
|
|
|
else { printPgmString(PSTR(" G18")); }
|
|
|
|
} else { printPgmString(PSTR(" G19")); }
|
|
|
|
|
|
|
|
if (gc.inches_mode) { printPgmString(PSTR(" G20")); }
|
|
|
|
else { printPgmString(PSTR(" G21")); }
|
|
|
|
|
|
|
|
if (gc.absolute_mode) { printPgmString(PSTR(" G90")); }
|
|
|
|
else { printPgmString(PSTR(" G91")); }
|
|
|
|
|
|
|
|
if (gc.inverse_feed_rate_mode) { printPgmString(PSTR(" G93")); }
|
|
|
|
else { printPgmString(PSTR(" G94")); }
|
|
|
|
|
|
|
|
// TODO: Check if G92 needs to be here. If so, how to track it.
|
|
|
|
|
|
|
|
switch (gc.program_flow) {
|
|
|
|
case PROGRAM_FLOW_RUNNING : printPgmString(PSTR(" M0")); break;
|
|
|
|
case PROGRAM_FLOW_PAUSED : printPgmString(PSTR(" M1")); break;
|
|
|
|
case PROGRAM_FLOW_COMPLETED : printPgmString(PSTR(" M2")); break;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (gc.spindle_direction) {
|
|
|
|
case 1 : printPgmString(PSTR(" M3")); break;
|
|
|
|
case -1 : printPgmString(PSTR(" M4")); break;
|
|
|
|
case 0 : printPgmString(PSTR(" M5")); break;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (gc.coolant_mode) {
|
|
|
|
case COOLANT_DISABLE : printPgmString(PSTR(" M9")); break;
|
|
|
|
case COOLANT_FLOOD_ENABLE : printPgmString(PSTR(" M8")); break;
|
|
|
|
// case COOLANT_MIST_ENABLE : printPgmString(PSTR(" M7")); break;
|
|
|
|
}
|
|
|
|
|
|
|
|
printPgmString(PSTR(" T"));
|
|
|
|
printInteger(gc.tool);
|
|
|
|
|
|
|
|
printPgmString(PSTR(" F"));
|
|
|
|
if (gc.inches_mode) { printFloat(gc.feed_rate*INCH_PER_MM); }
|
|
|
|
else { printFloat(gc.feed_rate); }
|
|
|
|
|
|
|
|
if (sys.switches) {
|
|
|
|
if (bit_istrue(sys.switches,BITFLAG_BLOCK_DELETE)) { printPgmString(PSTR(" $B")); }
|
|
|
|
if (bit_istrue(sys.switches,BITFLAG_SINGLE_BLOCK)) { printPgmString(PSTR(" $S")); }
|
|
|
|
if (bit_istrue(sys.switches,BITFLAG_OPT_STOP)) { printPgmString(PSTR(" $O")); }
|
|
|
|
}
|
|
|
|
|
|
|
|
printPgmString(PSTR("\r\n"));
|
|
|
|
}
|
2012-11-01 16:37:27 +01:00
|
|
|
|
|
|
|
|
|
|
|
void report_realtime_status()
|
|
|
|
{
|
|
|
|
// TODO: Status report data is written to the user here. This function should be able to grab a
|
|
|
|
// real-time snapshot of the stepper subprogram and the actual location of the CNC machine. At a
|
|
|
|
// minimum, status report should return real-time location information. Other important information
|
|
|
|
// may be distance to go on block, processed block id, and feed rate. A secondary, non-critical
|
|
|
|
// status report may include g-code state, i.e. inch mode, plane mode, absolute mode, etc.
|
|
|
|
// The report generated must be as short as possible, yet still provide the user easily readable
|
|
|
|
// information, i.e. '[0.23,120.4,2.4]'. This is necessary as it minimizes the computational
|
|
|
|
// overhead and allows grbl to keep running smoothly, especially with g-code programs with fast,
|
|
|
|
// short line segments and interface setups that require real-time status reports (5-20Hz).
|
|
|
|
|
|
|
|
// **Under construction** Bare-bones status report. Provides real-time machine position relative to
|
|
|
|
// the system power on location (0,0,0) and work coordinate position (G54 and G92 applied).
|
|
|
|
// The following are still needed: user setting of output units (mm|inch), compressed (non-human
|
|
|
|
// readable) data for interfaces?, save last known position in EEPROM?, code optimizations, solidify
|
|
|
|
// the reporting schemes, move to a separate .c file for easy user accessibility, and setting the
|
|
|
|
// home position by the user (likely through '$' setting interface).
|
|
|
|
// Successfully tested at a query rate of 10-20Hz while running a gauntlet of programs at various
|
|
|
|
// speeds.
|
|
|
|
uint8_t i;
|
|
|
|
int32_t current_position[3]; // Copy current state of the system position variable
|
|
|
|
memcpy(current_position,sys.position,sizeof(sys.position));
|
|
|
|
float print_position[3];
|
|
|
|
|
|
|
|
// Report machine position
|
|
|
|
printPgmString(PSTR("MPos:["));
|
|
|
|
for (i=0; i<= 2; i++) {
|
|
|
|
print_position[i] = current_position[i]/settings.steps_per_mm[i];
|
|
|
|
if (bit_istrue(settings.flags,BITFLAG_REPORT_INCHES)) { print_position[i] *= INCH_PER_MM; }
|
|
|
|
printFloat(print_position[i]);
|
|
|
|
if (i < 2) { printPgmString(PSTR(",")); }
|
|
|
|
else { printPgmString(PSTR("]")); }
|
|
|
|
}
|
|
|
|
|
|
|
|
// Report work position
|
|
|
|
printPgmString(PSTR(",WPos:["));
|
|
|
|
for (i=0; i<= 2; i++) {
|
|
|
|
if (bit_istrue(settings.flags,BITFLAG_REPORT_INCHES)) {
|
2012-11-02 02:48:55 +01:00
|
|
|
print_position[i] -= (gc.coord_system[i]+gc.coord_offset[i])*INCH_PER_MM;
|
2012-11-01 16:37:27 +01:00
|
|
|
} else {
|
2012-11-02 02:48:55 +01:00
|
|
|
print_position[i] -= gc.coord_system[i]+gc.coord_offset[i];
|
2012-11-01 16:37:27 +01:00
|
|
|
}
|
|
|
|
printFloat(print_position[i]);
|
|
|
|
if (i < 2) { printPgmString(PSTR(",")); }
|
|
|
|
else { printPgmString(PSTR("]")); }
|
|
|
|
}
|
|
|
|
|
|
|
|
printPgmString(PSTR("\r\n"));
|
|
|
|
}
|