mirror of
https://github.com/grblHAL/core.git
synced 2026-02-06 00:52:35 +08:00
226 lines
8.5 KiB
C
226 lines
8.5 KiB
C
/*
|
|
spindle_control.c - spindle control methods
|
|
|
|
Part of grblHAL
|
|
|
|
Copyright (c) 2017-2021 Terje Io
|
|
Copyright (c) 2012-2015 Sungeun K. Jeon
|
|
Copyright (c) 2009-2011 Simen Svale Skogsrud
|
|
|
|
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 <math.h>
|
|
|
|
#include "hal.h"
|
|
#include "protocol.h"
|
|
#include "state_machine.h"
|
|
|
|
// Set spindle speed override
|
|
// NOTE: Unlike motion overrides, spindle overrides do not require a planner reinitialization.
|
|
void spindle_set_override (uint_fast8_t speed_override)
|
|
{
|
|
if(sys.override.control.spindle_rpm_disable)
|
|
return;
|
|
|
|
speed_override = max(min(speed_override, MAX_SPINDLE_RPM_OVERRIDE), MIN_SPINDLE_RPM_OVERRIDE);
|
|
|
|
if ((uint8_t)speed_override != sys.override.spindle_rpm) {
|
|
sys.override.spindle_rpm = (uint8_t)speed_override;
|
|
if(state_get() == STATE_IDLE)
|
|
spindle_set_state(gc_state.modal.spindle, gc_state.spindle.rpm);
|
|
else
|
|
sys.step_control.update_spindle_rpm = On;
|
|
sys.report.overrides = On; // Set to report change immediately
|
|
}
|
|
}
|
|
|
|
// Immediately sets spindle running state with direction and spindle rpm, if enabled.
|
|
// Called by g-code parser spindle_sync(), parking retract and restore, g-code program end,
|
|
// sleep, and spindle stop override.
|
|
bool spindle_set_state (spindle_state_t state, float rpm)
|
|
{
|
|
if (!ABORTED) { // Block during abort.
|
|
|
|
if (!state.on) { // Halt or set spindle direction and rpm.
|
|
sys.spindle_rpm = rpm = 0.0f;
|
|
hal.spindle.set_state((spindle_state_t){0}, 0.0f);
|
|
} else {
|
|
// NOTE: Assumes all calls to this function is when Grbl is not moving or must remain off.
|
|
// TODO: alarm/interlock if going from CW to CCW directly in non-laser mode?
|
|
if (settings.mode == Mode_Laser && state.ccw)
|
|
rpm = 0.0f; // TODO: May need to be rpm_min*(100/MAX_SPINDLE_RPM_OVERRIDE);
|
|
|
|
hal.spindle.set_state(state, spindle_set_rpm(rpm, sys.override.spindle_rpm));
|
|
}
|
|
sys.report.spindle = On; // Set to report change immediately
|
|
|
|
st_rpm_changed(rpm);
|
|
}
|
|
|
|
return !ABORTED;
|
|
}
|
|
|
|
// G-code parser entry-point for setting spindle state. Forces a planner buffer sync and bails
|
|
// if an abort or check-mode is active.
|
|
bool spindle_sync (spindle_state_t state, float rpm)
|
|
{
|
|
bool ok = true;
|
|
bool at_speed = state_get() == STATE_CHECK_MODE || !state.on || !hal.driver_cap.spindle_at_speed || settings.spindle.at_speed_tolerance <= 0.0f;
|
|
|
|
if (state_get() != STATE_CHECK_MODE) {
|
|
// Empty planner buffer to ensure spindle is set when programmed.
|
|
if((ok = protocol_buffer_synchronize()) && spindle_set_state(state, rpm) && !at_speed) {
|
|
float delay = 0.0f;
|
|
while(!(at_speed = hal.spindle.get_state().at_speed)) {
|
|
delay_sec(0.1f, DelayMode_Dwell);
|
|
delay += 0.1f;
|
|
if(ABORTED)
|
|
break;
|
|
if(delay >= SAFETY_DOOR_SPINDLE_DELAY) {
|
|
system_raise_alarm(Alarm_Spindle);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return ok && at_speed;
|
|
}
|
|
|
|
// Restore spindle running state with direction, enable, spindle RPM and appropriate delay.
|
|
bool spindle_restore (spindle_state_t state, float rpm)
|
|
{
|
|
bool ok = true;
|
|
|
|
if(settings.mode == Mode_Laser) // When in laser mode, ignore spindle spin-up delay. Set to turn on laser when cycle starts.
|
|
sys.step_control.update_spindle_rpm = On;
|
|
else { // TODO: add check for current spindle state matches restore state?
|
|
spindle_set_state(state, rpm);
|
|
if(state.on) {
|
|
if((ok = !hal.driver_cap.spindle_at_speed))
|
|
delay_sec(SAFETY_DOOR_SPINDLE_DELAY, DelayMode_SysSuspend);
|
|
else if((ok == (settings.spindle.at_speed_tolerance <= 0.0f))) {
|
|
float delay = 0.0f;
|
|
while(!(ok = hal.spindle.get_state().at_speed)) {
|
|
delay_sec(0.1f, DelayMode_SysSuspend);
|
|
delay += 0.1f;
|
|
if(ABORTED)
|
|
break;
|
|
if(delay >= SAFETY_DOOR_SPINDLE_DELAY) {
|
|
system_raise_alarm(Alarm_Spindle);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return ok;
|
|
}
|
|
|
|
// Calculate and set programmed RPM according to override and max/min limits
|
|
float spindle_set_rpm (float rpm, uint8_t override_pct)
|
|
{
|
|
if(override_pct != 100)
|
|
rpm *= 0.01f * (float)override_pct; // Scale RPM by override value.
|
|
|
|
// Apply RPM limits
|
|
if (rpm <= 0.0f)
|
|
rpm = 0.0f;
|
|
else if (rpm > settings.spindle.rpm_max)
|
|
rpm = settings.spindle.rpm_max;
|
|
else if (rpm < settings.spindle.rpm_min)
|
|
rpm = settings.spindle.rpm_min;
|
|
|
|
sys.spindle_rpm = rpm;
|
|
|
|
return rpm;
|
|
}
|
|
|
|
//
|
|
// The following functions are not called by the core, may be called by driver code.
|
|
//
|
|
|
|
// calculate inverted pwm value if configured
|
|
static inline uint_fast16_t invert_pwm (spindle_pwm_t *pwm_data, uint_fast16_t pwm_value)
|
|
{
|
|
return pwm_data->invert_pwm ? pwm_data->period - pwm_value - 1 : pwm_value;
|
|
}
|
|
|
|
// Precompute PWM values for faster conversion.
|
|
// Returns false if no PWM range possible, driver should revert to simple on/off spindle control if so.
|
|
bool spindle_precompute_pwm_values (spindle_pwm_t *pwm_data, uint32_t clock_hz)
|
|
{
|
|
if(settings.spindle.rpm_max > settings.spindle.rpm_min) {
|
|
pwm_data->period = (uint_fast16_t)((float)clock_hz / settings.spindle.pwm_freq);
|
|
if(settings.spindle.pwm_off_value == 0.0f)
|
|
pwm_data->off_value = pwm_data->invert_pwm ? pwm_data->period : 0;
|
|
else
|
|
pwm_data->off_value = invert_pwm(pwm_data, (uint_fast16_t)(pwm_data->period * settings.spindle.pwm_off_value / 100.0f));
|
|
pwm_data->min_value = (uint_fast16_t)(pwm_data->period * settings.spindle.pwm_min_value / 100.0f);
|
|
pwm_data->max_value = (uint_fast16_t)(pwm_data->period * settings.spindle.pwm_max_value / 100.0f) + pwm_data->offset;
|
|
pwm_data->pwm_gradient = (float)(pwm_data->max_value - pwm_data->min_value) / (settings.spindle.rpm_max - settings.spindle.rpm_min);
|
|
pwm_data->always_on = settings.spindle.pwm_off_value != 0.0f;
|
|
}
|
|
|
|
#ifdef ENABLE_SPINDLE_LINEARIZATION
|
|
uint_fast8_t idx;
|
|
|
|
pwm_data->n_pieces = 0;
|
|
|
|
for(idx = 0; idx < SPINDLE_NPWM_PIECES; idx++) {
|
|
if(!isnan(settings.spindle.pwm_piece[idx].rpm) && settings.spindle.pwm_piece[idx].start != 0.0f)
|
|
memcpy(&pwm_data->piece[pwm_data->n_pieces++], &settings.spindle.pwm_piece[idx], sizeof(pwm_piece_t));
|
|
}
|
|
#endif
|
|
|
|
return settings.spindle.rpm_max > settings.spindle.rpm_min;
|
|
}
|
|
|
|
// Spindle RPM to PWM conversion.
|
|
uint_fast16_t spindle_compute_pwm_value (spindle_pwm_t *pwm_data, float rpm, bool pid_limit)
|
|
{
|
|
uint_fast16_t pwm_value;
|
|
|
|
if(rpm > settings.spindle.rpm_min) {
|
|
#ifdef ENABLE_SPINDLE_LINEARIZATION
|
|
// Compute intermediate PWM value with linear spindle speed model via piecewise linear fit model.
|
|
uint_fast8_t idx = pwm_data->n_pieces;
|
|
|
|
if(idx) {
|
|
do {
|
|
idx--;
|
|
if(idx == 0 || rpm > pwm_data->piece[idx].rpm) {
|
|
pwm_value = floorf(pwm_data->piece[idx].start * rpm - pwm_data->piece[idx].end);
|
|
break;
|
|
}
|
|
} while(idx);
|
|
} else
|
|
#endif
|
|
// Compute intermediate PWM value with linear spindle speed model.
|
|
pwm_value = (uint_fast16_t)floorf((rpm - settings.spindle.rpm_min) * pwm_data->pwm_gradient) + pwm_data->min_value;
|
|
|
|
if(pwm_value >= (pid_limit ? pwm_data->period : pwm_data->max_value))
|
|
pwm_value = pid_limit ? pwm_data->period - 1 : pwm_data->max_value;
|
|
else if(pwm_value < pwm_data->min_value)
|
|
pwm_value = pwm_data->min_value;
|
|
|
|
pwm_value = invert_pwm(pwm_data, pwm_value);
|
|
} else
|
|
pwm_value = rpm == 0.0f ? pwm_data->off_value : invert_pwm(pwm_data, pwm_data->min_value);
|
|
|
|
return pwm_value;
|
|
}
|