2010-03-07 20:29:18 +01:00
|
|
|
/*
|
2011-02-05 00:55:37 +01:00
|
|
|
settings.c - eeprom configuration handling
|
2010-03-07 20:29:18 +01:00
|
|
|
Part of Grbl
|
|
|
|
|
2011-01-14 16:45:18 +01:00
|
|
|
Copyright (c) 2009-2011 Simen Svale Skogsrud
|
2012-10-10 06:01:10 +02:00
|
|
|
Copyright (c) 2011-2012 Sungeun K. Jeon
|
2010-03-07 20:29:18 +01:00
|
|
|
|
|
|
|
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/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <avr/io.h>
|
|
|
|
#include <math.h>
|
|
|
|
#include "nuts_bolts.h"
|
2011-02-05 00:45:41 +01:00
|
|
|
#include "settings.h"
|
2010-03-07 20:29:18 +01:00
|
|
|
#include "eeprom.h"
|
2011-06-03 15:28:14 +02:00
|
|
|
#include "print.h"
|
2010-03-07 23:10:41 +01:00
|
|
|
#include <avr/pgmspace.h>
|
2011-02-18 22:59:16 +01:00
|
|
|
#include "protocol.h"
|
2011-05-31 13:13:45 +02:00
|
|
|
#include "config.h"
|
2010-03-07 20:29:18 +01:00
|
|
|
|
2011-01-25 22:51:37 +01:00
|
|
|
settings_t settings;
|
2011-01-15 00:27:08 +01:00
|
|
|
|
2011-01-31 23:04:08 +01:00
|
|
|
// Version 1 outdated settings record
|
|
|
|
typedef struct {
|
2012-10-08 23:57:58 +02:00
|
|
|
float steps_per_mm[3];
|
2011-01-31 23:04:08 +01:00
|
|
|
uint8_t microsteps;
|
|
|
|
uint8_t pulse_microseconds;
|
2012-10-08 23:57:58 +02:00
|
|
|
float default_feed_rate;
|
|
|
|
float default_seek_rate;
|
2011-01-31 23:04:08 +01:00
|
|
|
uint8_t invert_mask;
|
2012-10-08 23:57:58 +02:00
|
|
|
float mm_per_arc_segment;
|
2011-01-31 23:04:08 +01:00
|
|
|
} settings_v1_t;
|
|
|
|
|
2012-10-10 06:01:10 +02:00
|
|
|
// Version 2,3,4 outdated settings record
|
|
|
|
typedef struct {
|
|
|
|
float steps_per_mm[3];
|
|
|
|
uint8_t microsteps;
|
|
|
|
uint8_t pulse_microseconds;
|
|
|
|
float default_feed_rate;
|
|
|
|
float default_seek_rate;
|
|
|
|
uint8_t invert_mask;
|
|
|
|
float mm_per_arc_segment;
|
|
|
|
float acceleration;
|
|
|
|
float junction_deviation;
|
|
|
|
} settings_v2_v4_t;
|
|
|
|
|
2011-02-17 20:23:12 +01:00
|
|
|
// Default settings (used when resetting eeprom-settings)
|
|
|
|
#define MICROSTEPS 8
|
|
|
|
#define DEFAULT_X_STEPS_PER_MM (94.488188976378*MICROSTEPS)
|
|
|
|
#define DEFAULT_Y_STEPS_PER_MM (94.488188976378*MICROSTEPS)
|
|
|
|
#define DEFAULT_Z_STEPS_PER_MM (94.488188976378*MICROSTEPS)
|
|
|
|
#define DEFAULT_STEP_PULSE_MICROSECONDS 30
|
|
|
|
#define DEFAULT_MM_PER_ARC_SEGMENT 0.1
|
2011-10-12 04:51:04 +02:00
|
|
|
#define DEFAULT_RAPID_FEEDRATE 500.0 // mm/min
|
2011-02-17 20:24:04 +01:00
|
|
|
#define DEFAULT_FEEDRATE 500.0
|
2011-10-12 04:51:04 +02:00
|
|
|
#define DEFAULT_ACCELERATION (DEFAULT_FEEDRATE*60*60/10.0) // mm/min^2
|
|
|
|
#define DEFAULT_JUNCTION_DEVIATION 0.05 // mm
|
2011-06-05 21:34:53 +02:00
|
|
|
#define DEFAULT_STEPPING_INVERT_MASK ((1<<X_STEP_BIT)|(1<<Y_STEP_BIT)|(1<<Z_STEP_BIT))
|
2012-10-10 06:01:10 +02:00
|
|
|
|
|
|
|
// Developmental default settings
|
|
|
|
#define DEFAULT_HOMING_ENABLE 0 // false
|
|
|
|
#define DEFAULT_HOMING_RAPID_FEEDRATE 250.0 // mm/min
|
|
|
|
#define DEFAULT_HOMING_FEEDRATE 50 // mm/min
|
|
|
|
#define DEFAULT_HOMING_DEBOUNCE_DELAY 100 // msec (0-65k)
|
|
|
|
#define DEFAULT_STEPPER_IDLE_LOCK_TIME 25 // msec (0-255)
|
|
|
|
// #define DEFAULT_AUTO_START 1 // true
|
2012-10-11 02:01:43 +02:00
|
|
|
// #define DEFAULT_INCHES_MODE 1 // true
|
2012-10-10 06:01:10 +02:00
|
|
|
// #define DEFAULT_BLOCK_DELETE 0 // false
|
2012-10-12 06:43:54 +02:00
|
|
|
#define DEFAULT_DECIMAL_PLACES 3
|
2011-02-17 20:23:12 +01:00
|
|
|
|
2012-10-11 02:01:43 +02:00
|
|
|
void settings_reset(bool reset_all) {
|
|
|
|
// Reset all settings or only the migration settings to the new version.
|
|
|
|
if (reset_all) {
|
|
|
|
settings.steps_per_mm[X_AXIS] = DEFAULT_X_STEPS_PER_MM;
|
|
|
|
settings.steps_per_mm[Y_AXIS] = DEFAULT_Y_STEPS_PER_MM;
|
|
|
|
settings.steps_per_mm[Z_AXIS] = DEFAULT_Z_STEPS_PER_MM;
|
|
|
|
settings.pulse_microseconds = DEFAULT_STEP_PULSE_MICROSECONDS;
|
|
|
|
settings.default_feed_rate = DEFAULT_FEEDRATE;
|
|
|
|
settings.default_seek_rate = DEFAULT_RAPID_FEEDRATE;
|
|
|
|
settings.acceleration = DEFAULT_ACCELERATION;
|
|
|
|
settings.mm_per_arc_segment = DEFAULT_MM_PER_ARC_SEGMENT;
|
|
|
|
settings.invert_mask = DEFAULT_STEPPING_INVERT_MASK;
|
|
|
|
settings.junction_deviation = DEFAULT_JUNCTION_DEVIATION;
|
|
|
|
}
|
|
|
|
// New settings since last version
|
|
|
|
settings.flags = 0;
|
|
|
|
// if (DEFAULT_AUTO_START) { settings.flags |= FLAG_BIT_AUTO_START; }
|
|
|
|
// if (DEFAULT_INCHES_MODE) { settings.flags |= FLAG_BIT_INCHES_MODE; }
|
|
|
|
if (DEFAULT_HOMING_ENABLE) { settings.flags |= FLAG_BIT_HOMING_ENABLE; }
|
|
|
|
settings.homing_feed_rate = DEFAULT_HOMING_FEEDRATE;
|
|
|
|
settings.homing_seek_rate = DEFAULT_HOMING_RAPID_FEEDRATE;
|
|
|
|
settings.homing_debounce_delay = DEFAULT_HOMING_DEBOUNCE_DELAY;
|
|
|
|
settings.stepper_idle_lock_time = DEFAULT_STEPPER_IDLE_LOCK_TIME;
|
2012-10-12 06:43:54 +02:00
|
|
|
settings.decimal_places = DEFAULT_DECIMAL_PLACES;
|
2010-03-07 20:29:18 +01:00
|
|
|
}
|
|
|
|
|
2012-10-13 21:11:43 +02:00
|
|
|
// static void settings_startup_string(char *buf) {
|
|
|
|
// memcpy_from_eeprom_with_checksum((char*)buf,512, 4);
|
|
|
|
// }
|
|
|
|
|
2011-02-05 01:00:41 +01:00
|
|
|
void settings_dump() {
|
2011-02-06 23:27:04 +01:00
|
|
|
printPgmString(PSTR("$0 = ")); printFloat(settings.steps_per_mm[X_AXIS]);
|
|
|
|
printPgmString(PSTR(" (steps/mm x)\r\n$1 = ")); printFloat(settings.steps_per_mm[Y_AXIS]);
|
|
|
|
printPgmString(PSTR(" (steps/mm y)\r\n$2 = ")); printFloat(settings.steps_per_mm[Z_AXIS]);
|
2010-03-07 23:10:41 +01:00
|
|
|
printPgmString(PSTR(" (steps/mm z)\r\n$3 = ")); printInteger(settings.pulse_microseconds);
|
|
|
|
printPgmString(PSTR(" (microseconds step pulse)\r\n$4 = ")); printFloat(settings.default_feed_rate);
|
2011-01-14 12:10:18 +01:00
|
|
|
printPgmString(PSTR(" (mm/min default feed rate)\r\n$5 = ")); printFloat(settings.default_seek_rate);
|
|
|
|
printPgmString(PSTR(" (mm/min default seek rate)\r\n$6 = ")); printFloat(settings.mm_per_arc_segment);
|
2011-01-22 23:29:02 +01:00
|
|
|
printPgmString(PSTR(" (mm/arc segment)\r\n$7 = ")); printInteger(settings.invert_mask);
|
2012-01-06 18:10:41 +01:00
|
|
|
printPgmString(PSTR(" (step port invert mask. binary = ")); print_uint8_base2(settings.invert_mask);
|
2011-10-12 04:51:04 +02:00
|
|
|
printPgmString(PSTR(")\r\n$8 = ")); printFloat(settings.acceleration/(60*60)); // Convert from mm/min^2 for human readability
|
2011-09-03 23:31:48 +02:00
|
|
|
printPgmString(PSTR(" (acceleration in mm/sec^2)\r\n$9 = ")); printFloat(settings.junction_deviation);
|
2012-10-10 06:01:10 +02:00
|
|
|
printPgmString(PSTR(" (cornering junction deviation in mm)\r\n$10 = ")); printInteger(bit_istrue(settings.flags,FLAG_BIT_HOMING_ENABLE));
|
|
|
|
printPgmString(PSTR(" (boolean homing enable)\r\n$11 = ")); printFloat(settings.homing_feed_rate);
|
|
|
|
printPgmString(PSTR(" (mm/min homing feed rate)\r\n$12 = ")); printFloat(settings.homing_seek_rate);
|
|
|
|
printPgmString(PSTR(" (mm/min homing seek rate)\r\n$13 = ")); printInteger(settings.homing_debounce_delay);
|
|
|
|
printPgmString(PSTR(" (milliseconds homing debounce delay)\r\n$14 = ")); printInteger(settings.stepper_idle_lock_time);
|
2012-10-12 06:43:54 +02:00
|
|
|
printPgmString(PSTR(" (milliseconds stepper idle lock time)\r\n$15 = ")); printInteger(settings.decimal_places);
|
|
|
|
printPgmString(PSTR(" (float decimal places)"));
|
2012-10-13 21:11:43 +02:00
|
|
|
|
|
|
|
// char buf[4];
|
|
|
|
// settings_startup_string((char *)buf);
|
|
|
|
// printPgmString(PSTR("\r\n Startup: ")); printString(buf);
|
|
|
|
|
2010-12-20 14:01:38 +01:00
|
|
|
printPgmString(PSTR("\r\n'$x=value' to set parameter or just '$' to dump current settings\r\n"));
|
2010-03-07 20:29:18 +01:00
|
|
|
}
|
|
|
|
|
2011-02-18 22:59:16 +01:00
|
|
|
// Parameter lines are on the form '$4=374.3' or '$' to dump current settings
|
|
|
|
uint8_t settings_execute_line(char *line) {
|
|
|
|
uint8_t char_counter = 1;
|
2012-10-13 21:11:43 +02:00
|
|
|
// unsigned char letter;
|
2012-10-08 23:57:58 +02:00
|
|
|
float parameter, value;
|
2011-02-18 22:59:16 +01:00
|
|
|
if(line[0] != '$') {
|
|
|
|
return(STATUS_UNSUPPORTED_STATEMENT);
|
|
|
|
}
|
|
|
|
if(line[char_counter] == 0) {
|
|
|
|
settings_dump(); return(STATUS_OK);
|
|
|
|
}
|
2012-10-13 21:11:43 +02:00
|
|
|
// if(line[char_counter] >= 'A' || line[char_counter] <= 'Z') {
|
|
|
|
// letter = line[char_counter++];
|
|
|
|
// if(line[char_counter++] != '=') {
|
|
|
|
// return(STATUS_UNSUPPORTED_STATEMENT);
|
|
|
|
// }
|
|
|
|
// for (char_counter = 0; char_counter < LINE_BUFFER_SIZE-3; char_counter++) {
|
|
|
|
// line[char_counter] = line[char_counter+3];
|
|
|
|
// }
|
|
|
|
// uint8_t status = gc_execute_line(line);
|
|
|
|
// if (status) { return(status); }
|
|
|
|
// else { settings_store_startup_line(line); }
|
|
|
|
//
|
|
|
|
//
|
|
|
|
// // Opt stop and block delete are referred to as switches.
|
|
|
|
// // How to store home position and work offsets real-time??
|
|
|
|
//
|
|
|
|
// } else {
|
2012-10-08 23:57:58 +02:00
|
|
|
if(!read_float(line, &char_counter, ¶meter)) {
|
2011-02-18 23:08:06 +01:00
|
|
|
return(STATUS_BAD_NUMBER_FORMAT);
|
|
|
|
};
|
2011-02-18 22:59:16 +01:00
|
|
|
if(line[char_counter++] != '=') {
|
|
|
|
return(STATUS_UNSUPPORTED_STATEMENT);
|
|
|
|
}
|
2012-10-08 23:57:58 +02:00
|
|
|
if(!read_float(line, &char_counter, &value)) {
|
2011-02-18 23:08:06 +01:00
|
|
|
return(STATUS_BAD_NUMBER_FORMAT);
|
|
|
|
}
|
2011-02-18 22:59:16 +01:00
|
|
|
if(line[char_counter] != 0) {
|
|
|
|
return(STATUS_UNSUPPORTED_STATEMENT);
|
|
|
|
}
|
|
|
|
settings_store_setting(parameter, value);
|
|
|
|
return(STATUS_OK);
|
2012-10-13 21:11:43 +02:00
|
|
|
// }
|
2011-02-18 22:59:16 +01:00
|
|
|
}
|
|
|
|
|
2011-01-31 23:04:08 +01:00
|
|
|
void write_settings() {
|
|
|
|
eeprom_put_char(0, SETTINGS_VERSION);
|
|
|
|
memcpy_to_eeprom_with_checksum(1, (char*)&settings, sizeof(settings_t));
|
2012-10-13 21:11:43 +02:00
|
|
|
//
|
|
|
|
// char buf[4]; buf[0] = 'G'; buf[1] = '2'; buf[2] = '0'; buf[3] = 0;
|
|
|
|
// memcpy_to_eeprom_with_checksum(512, (char*)buf, 4);
|
|
|
|
//
|
2011-01-31 23:04:08 +01:00
|
|
|
}
|
|
|
|
|
2010-03-07 20:29:18 +01:00
|
|
|
int read_settings() {
|
|
|
|
// Check version-byte of eeprom
|
2011-01-24 23:32:33 +01:00
|
|
|
uint8_t version = eeprom_get_char(0);
|
2011-01-31 23:04:08 +01:00
|
|
|
|
|
|
|
if (version == SETTINGS_VERSION) {
|
|
|
|
// Read settings-record and check checksum
|
|
|
|
if (!(memcpy_from_eeprom_with_checksum((char*)&settings, 1, sizeof(settings_t)))) {
|
2011-02-18 23:04:12 +01:00
|
|
|
return(false);
|
2011-01-31 23:04:08 +01:00
|
|
|
}
|
2012-10-10 06:01:10 +02:00
|
|
|
} else {
|
|
|
|
// Incrementally update the old versions until up-to-date.
|
|
|
|
if (version == 1) {
|
|
|
|
// Migrate from settings version 1 to version 4.
|
|
|
|
if (!(memcpy_from_eeprom_with_checksum((char*)&settings, 1, sizeof(settings_v1_t)))) {
|
|
|
|
return(false);
|
|
|
|
}
|
|
|
|
settings.acceleration = DEFAULT_ACCELERATION;
|
|
|
|
settings.junction_deviation = DEFAULT_JUNCTION_DEVIATION;
|
|
|
|
} else if ((version == 2) || (version == 3)) {
|
|
|
|
// Migrate from settings version 2 and 3 to version 4.
|
|
|
|
if (!(memcpy_from_eeprom_with_checksum((char*)&settings, 1, sizeof(settings_v2_v4_t)))) {
|
|
|
|
return(false);
|
|
|
|
}
|
|
|
|
if (version == 2) { settings.junction_deviation = DEFAULT_JUNCTION_DEVIATION; }
|
|
|
|
settings.acceleration *= 3600; // Convert to mm/min^2 from mm/sec^2
|
|
|
|
}
|
|
|
|
if (version <= 4) {
|
|
|
|
// Migrate from settings version 4 to current version.
|
|
|
|
if (!(memcpy_from_eeprom_with_checksum((char*)&settings, 1, sizeof(settings_v2_v4_t)))) {
|
|
|
|
return(false);
|
2012-10-11 02:01:43 +02:00
|
|
|
}
|
|
|
|
settings_reset(false);
|
2012-10-10 06:01:10 +02:00
|
|
|
write_settings();
|
|
|
|
} else if (version >= 50) {
|
|
|
|
// Developmental settings. Version numbers greater than or equal to 50 are temporary.
|
|
|
|
// Currently, this will update the user settings to v4 and the remainder of the settings
|
|
|
|
// should be re-written to the default value, if the developmental version number changed.
|
|
|
|
|
|
|
|
// Grab settings regardless of error.
|
|
|
|
memcpy_from_eeprom_with_checksum((char*)&settings, 1, sizeof(settings_t));
|
2012-10-11 02:01:43 +02:00
|
|
|
settings_reset(false);
|
2012-10-10 06:01:10 +02:00
|
|
|
write_settings();
|
|
|
|
} else {
|
2011-09-24 15:46:41 +02:00
|
|
|
return(false);
|
|
|
|
}
|
2010-03-07 20:29:18 +01:00
|
|
|
}
|
2011-02-18 23:04:12 +01:00
|
|
|
return(true);
|
2010-03-07 20:29:18 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// A helper method to set settings from command line
|
2012-10-08 23:57:58 +02:00
|
|
|
void settings_store_setting(int parameter, float value) {
|
2010-03-07 20:29:18 +01:00
|
|
|
switch(parameter) {
|
|
|
|
case 0: case 1: case 2:
|
2011-02-25 13:40:48 +01:00
|
|
|
if (value <= 0.0) {
|
|
|
|
printPgmString(PSTR("Steps/mm must be > 0.0\r\n"));
|
|
|
|
return;
|
|
|
|
}
|
2010-03-07 20:29:18 +01:00
|
|
|
settings.steps_per_mm[parameter] = value; break;
|
2011-07-17 19:35:44 +02:00
|
|
|
case 3:
|
|
|
|
if (value < 3) {
|
|
|
|
printPgmString(PSTR("Step pulse must be >= 3 microseconds\r\n"));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
settings.pulse_microseconds = round(value); break;
|
2010-03-07 20:29:18 +01:00
|
|
|
case 4: settings.default_feed_rate = value; break;
|
|
|
|
case 5: settings.default_seek_rate = value; break;
|
2010-12-20 14:01:38 +01:00
|
|
|
case 6: settings.mm_per_arc_segment = value; break;
|
2011-01-22 23:29:02 +01:00
|
|
|
case 7: settings.invert_mask = trunc(value); break;
|
2011-10-12 04:51:04 +02:00
|
|
|
case 8: settings.acceleration = value*60*60; break; // Convert to mm/min^2 for grbl internal use.
|
2011-09-03 23:31:48 +02:00
|
|
|
case 9: settings.junction_deviation = fabs(value); break;
|
2012-10-10 06:01:10 +02:00
|
|
|
case 10:
|
|
|
|
if (value) {
|
|
|
|
settings.flags |= FLAG_BIT_HOMING_ENABLE;
|
|
|
|
printPgmString(PSTR("Install all axes limit switches before use\r\n"));
|
|
|
|
} else { settings.flags &= ~FLAG_BIT_HOMING_ENABLE; }
|
|
|
|
break;
|
|
|
|
case 11: settings.homing_feed_rate = value; break;
|
|
|
|
case 12: settings.homing_seek_rate = value; break;
|
|
|
|
case 13: settings.homing_debounce_delay = round(value); break;
|
|
|
|
case 14: settings.stepper_idle_lock_time = round(value); break;
|
2012-10-12 06:43:54 +02:00
|
|
|
case 15: settings.decimal_places = round(value); break;
|
2010-03-07 20:29:18 +01:00
|
|
|
default:
|
2011-01-14 12:10:18 +01:00
|
|
|
printPgmString(PSTR("Unknown parameter\r\n"));
|
|
|
|
return;
|
2010-03-07 20:29:18 +01:00
|
|
|
}
|
|
|
|
write_settings();
|
2010-03-07 23:10:41 +01:00
|
|
|
printPgmString(PSTR("Stored new setting\r\n"));
|
2010-03-07 20:29:18 +01:00
|
|
|
}
|
|
|
|
|
2011-01-31 23:04:08 +01:00
|
|
|
// Initialize the config subsystem
|
2011-02-05 00:45:41 +01:00
|
|
|
void settings_init() {
|
2011-12-09 02:47:48 +01:00
|
|
|
if(!read_settings()) {
|
2010-04-22 20:43:37 +02:00
|
|
|
printPgmString(PSTR("Warning: Failed to read EEPROM settings. Using defaults.\r\n"));
|
2012-10-11 02:01:43 +02:00
|
|
|
settings_reset(true);
|
2010-03-07 20:29:18 +01:00
|
|
|
write_settings();
|
2011-02-05 01:00:41 +01:00
|
|
|
settings_dump();
|
2012-10-13 21:11:43 +02:00
|
|
|
}
|
2010-03-07 20:29:18 +01:00
|
|
|
}
|
2012-10-13 21:11:43 +02:00
|
|
|
|
|
|
|
// int8_t settings_execute_startup() {
|
|
|
|
//
|
|
|
|
// char buf[4];
|
|
|
|
// settings_startup_string((char *)buf);
|
|
|
|
// uint8_t i = 0;
|
|
|
|
// while (i < 4) {
|
|
|
|
// serial_write(buf[i++]);
|
|
|
|
// }
|
|
|
|
// return(gc_execute_line(buf));
|
|
|
|
// }
|