171 lines
7.1 KiB
C++
171 lines
7.1 KiB
C++
/// @file unload_filament.cpp
|
|
#include "unload_filament.h"
|
|
#include "../modules/finda.h"
|
|
#include "../modules/fsensor.h"
|
|
#include "../modules/globals.h"
|
|
#include "../modules/idler.h"
|
|
#include "../modules/leds.h"
|
|
#include "../modules/motion.h"
|
|
#include "../modules/permanent_storage.h"
|
|
#include "../modules/pulley.h"
|
|
#include "../modules/selector.h"
|
|
#include "../modules/user_input.h"
|
|
#include "../debug.h"
|
|
|
|
namespace logic {
|
|
|
|
UnloadFilament unloadFilament;
|
|
|
|
bool UnloadFilament::Reset(uint8_t /*param*/) {
|
|
|
|
if (!mf::finda.Pressed() && mg::globals.FilamentLoaded() < mg::FilamentLoadState::InSelector) {
|
|
// it looks like we have nothing in the PTFE tube, at least FINDA doesn't sense anything
|
|
// so the filament has been probably already unloaded - terminate with OK or report an error?
|
|
return true;
|
|
}
|
|
|
|
// unloads filament from extruder - filament is above Bondtech gears
|
|
mpu::pulley.InitAxis();
|
|
state = ProgressCode::UnloadingToFinda;
|
|
error = ErrorCode::RUNNING;
|
|
unl.Reset(maxRetries);
|
|
ml::leds.SetPairButOffOthers(mg::globals.ActiveSlot(), ml::off, ml::off);
|
|
return true;
|
|
}
|
|
|
|
void UnloadFilament::UnloadFinishedCorrectly() {
|
|
FinishedOK();
|
|
mpu::pulley.Disable();
|
|
mg::globals.SetFilamentLoaded(mg::globals.ActiveSlot(), mg::FilamentLoadState::AtPulley); // filament unloaded
|
|
ml::leds.SetPairButOffOthers(mg::globals.ActiveSlot(), ml::off, ml::off);
|
|
}
|
|
|
|
void UnloadFilament::GoToRetractingFromFinda() {
|
|
state = ProgressCode::RetractingFromFinda;
|
|
retract.Reset();
|
|
}
|
|
|
|
void UnloadFilament::GoToRecheckFilamentAgainstFINDA() {
|
|
state = ProgressCode::FeedingToFinda;
|
|
error = ErrorCode::RUNNING;
|
|
feed.Reset(true, true);
|
|
}
|
|
|
|
bool UnloadFilament::StepInner() {
|
|
switch (state) {
|
|
// state 1 engage idler - will be done by the Unload to FINDA state machine
|
|
case ProgressCode::UnloadingToFinda: // state 2 rotate pulley as long as the FINDA is on
|
|
if (unl.Step()) {
|
|
if (unl.State() == UnloadToFinda::FailedFINDA) {
|
|
// couldn't unload to FINDA, report error and wait for user to resolve it
|
|
GoToErrDisengagingIdler(ErrorCode::FINDA_DIDNT_SWITCH_OFF);
|
|
} else if (unl.State() == UnloadToFinda::FailedFSensor) {
|
|
// fsensor still pressed - that smells bad - a piece of filament may still be present in the heatsink
|
|
// and that would cause serious problems while loading another filament
|
|
GoToErrDisengagingIdler(ErrorCode::FSENSOR_DIDNT_SWITCH_OFF);
|
|
} else {
|
|
GoToRetractingFromFinda();
|
|
}
|
|
}
|
|
return false;
|
|
case ProgressCode::RetractingFromFinda:
|
|
if (retract.Step()) {
|
|
if (retract.State() == RetractFromFinda::Failed) {
|
|
GoToErrDisengagingIdler(ErrorCode::FINDA_DIDNT_SWITCH_OFF); // signal unloading error
|
|
} else {
|
|
state = ProgressCode::DisengagingIdler;
|
|
mi::idler.Disengage();
|
|
}
|
|
}
|
|
return false;
|
|
case ProgressCode::DisengagingIdler:
|
|
if (mi::idler.Disengaged() && ms::selector.State() == ms::Selector::Ready) {
|
|
UnloadFinishedCorrectly();
|
|
}
|
|
return false;
|
|
case ProgressCode::ERRDisengagingIdler: // couldn't unload to FINDA
|
|
ErrDisengagingIdler();
|
|
return false;
|
|
case ProgressCode::ERRWaitingForUser: {
|
|
// waiting for user buttons and/or a command from the printer
|
|
mui::Event ev = mui::userInput.ConsumeEvent();
|
|
switch (ev) {
|
|
case mui::Event::Left: // try to manually unload just a tiny bit - help the filament with the pulley
|
|
GoToErrEngagingIdler();
|
|
break;
|
|
case mui::Event::Middle: // try again the whole sequence
|
|
// First invalidate homing flags as the user may have moved the Idler or Selector accidentally
|
|
InvalidateHoming();
|
|
if (mf::finda.Pressed()) {
|
|
Reset(0); // filament is present in FINDA (regardless of FSensor) - assume we need to pull the filament to FINDA first
|
|
} else if (!mf::finda.Pressed() && mfs::fsensor.Pressed()) {
|
|
// a piece of filament is stuck in the extruder - keep waiting for the user to fix it
|
|
} else {
|
|
// filament is not present in FINDA and not in FSensor
|
|
// - that means the filament can still be behind FINDA and blocking the selector
|
|
// Ideally push it to FINDA and then back to verify the whole situation
|
|
GoToRecheckFilamentAgainstFINDA();
|
|
}
|
|
break;
|
|
case mui::Event::Right: // problem resolved - the user pulled the fillament by hand
|
|
// we should check the state of all the sensors and either report another error or confirm the correct state
|
|
|
|
// First invalidate homing flags as the user may have moved the Idler or Selector accidentally
|
|
InvalidateHoming();
|
|
if (mfs::fsensor.Pressed()) {
|
|
// printer's filament sensor is still pressed - that smells bad
|
|
error = ErrorCode::FSENSOR_DIDNT_SWITCH_OFF;
|
|
state = ProgressCode::ERRWaitingForUser; // stand still
|
|
} else if (mf::finda.Pressed()) {
|
|
// FINDA is still pressed - that smells bad
|
|
error = ErrorCode::FINDA_DIDNT_SWITCH_OFF;
|
|
state = ProgressCode::ERRWaitingForUser; // stand still
|
|
} else {
|
|
// all sensors are ok, but re-check the position of the filament against FINDA
|
|
GoToRecheckFilamentAgainstFINDA();
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
return false;
|
|
}
|
|
case ProgressCode::ERREngagingIdler:
|
|
if (mi::idler.Engaged()) {
|
|
state = ProgressCode::ERRHelpingFilament;
|
|
mpu::pulley.PlanMove(-config::pulleyHelperMove, config::pulleySlowFeedrate);
|
|
}
|
|
return false;
|
|
case ProgressCode::ERRHelpingFilament:
|
|
if (!mf::finda.Pressed()) {
|
|
// the help was enough to depress the FINDA, we are ok, continue normally
|
|
state = ProgressCode::DisengagingIdler;
|
|
error = ErrorCode::RUNNING;
|
|
} else if (mm::motion.QueueEmpty()) {
|
|
// helped a bit, but FINDA didn't trigger, return to the main error state
|
|
GoToErrDisengagingIdler(ErrorCode::FINDA_DIDNT_SWITCH_OFF);
|
|
}
|
|
return false;
|
|
case ProgressCode::FeedingToFinda:
|
|
// recovery mode - we assume the filament is somewhere between the idle position and FINDA - thus blocking the selector
|
|
if (feed.Step()) {
|
|
if (feed.State() == FeedToFinda::Failed) {
|
|
GoToErrDisengagingIdler(ErrorCode::FINDA_DIDNT_SWITCH_ON);
|
|
} else {
|
|
state = ProgressCode::RetractingFromFinda;
|
|
retract.Reset();
|
|
}
|
|
}
|
|
break;
|
|
case ProgressCode::OK:
|
|
return true; // successfully finished
|
|
default: // we got into an unhandled state, better report it
|
|
state = ProgressCode::ERRInternal;
|
|
error = ErrorCode::INTERNAL;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
} // namespace logic
|