83 lines
3.3 KiB
C++
83 lines
3.3 KiB
C++
/// @file unload_to_finda.cpp
|
|
#include "unload_to_finda.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"
|
|
|
|
namespace logic {
|
|
|
|
void UnloadToFinda::Reset(uint8_t maxTries) {
|
|
this->maxTries = maxTries;
|
|
// check the inital state of FINDA and plan the moves
|
|
if (!mf::finda.Pressed()) {
|
|
state = OK; // FINDA is already off, we assume the fillament is not there, i.e. already unloaded
|
|
} else {
|
|
// FINDA is sensing the filament, plan moves to unload it
|
|
state = EngagingIdler;
|
|
mi::idler.Engage(mg::globals.ActiveSlot());
|
|
}
|
|
}
|
|
|
|
// @@TODO this may end up somewhere else as more code may need to check the distance traveled by the filament
|
|
int32_t CurrentPositionPulley_mm() {
|
|
return mm::axisUnitToTruncatedUnit<config::U_mm>(mm::motion.CurPosition<mm::Pulley>());
|
|
}
|
|
|
|
bool UnloadToFinda::Step() {
|
|
switch (state) {
|
|
case EngagingIdler:
|
|
if (mg::globals.FilamentLoaded() >= mg::FilamentLoadState::InSelector) {
|
|
state = UnloadingToFinda;
|
|
mpu::pulley.InitAxis();
|
|
ml::leds.SetMode(mg::globals.ActiveSlot(), ml::green, ml::blink0);
|
|
} else {
|
|
state = FailedFINDA;
|
|
}
|
|
return false;
|
|
case UnloadingToFinda:
|
|
if (mi::idler.Engaged()) {
|
|
state = WaitingForFINDA;
|
|
mg::globals.SetFilamentLoaded(mg::globals.ActiveSlot(), mg::FilamentLoadState::InSelector);
|
|
unloadStart_mm = mpu::pulley.CurrentPositionPulley_mm();
|
|
mpu::pulley.PlanMove(-config::defaultBowdenLength - config::feedToFinda - config::filamentMinLoadedToMMU, config::pulleyUnloadFeedrate);
|
|
}
|
|
return false;
|
|
case WaitingForFINDA: {
|
|
int32_t currentPulley_mm = CurrentPositionPulley_mm();
|
|
if ((abs(unloadStart_mm - currentPulley_mm) > mm::truncatedUnit(config::fsensorUnloadCheckDistance)) && mfs::fsensor.Pressed()) {
|
|
// fsensor didn't trigger within the first fsensorUnloadCheckDistance mm -> stop pulling, something failed, report an error
|
|
// This scenario should not be tried again - repeating it may cause more damage to filament + potentially more collateral damage
|
|
state = FailedFSensor;
|
|
mm::motion.AbortPlannedMoves(); // stop rotating the pulley
|
|
ml::leds.SetMode(mg::globals.ActiveSlot(), ml::green, ml::off);
|
|
} else if (!mf::finda.Pressed()) {
|
|
// detected end of filament
|
|
state = OK;
|
|
mm::motion.AbortPlannedMoves(); // stop rotating the pulley
|
|
ml::leds.SetMode(mg::globals.ActiveSlot(), ml::green, ml::off);
|
|
} else if (/*tmc2130_read_gstat() &&*/ mm::motion.QueueEmpty()) {
|
|
// we reached the end of move queue, but the FINDA didn't switch off
|
|
// two possible causes - grinded filament or malfunctioning FINDA
|
|
if (--maxTries) {
|
|
Reset(maxTries); // try again
|
|
} else {
|
|
state = FailedFINDA;
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
case OK:
|
|
case FailedFINDA:
|
|
case FailedFSensor:
|
|
default:
|
|
return true;
|
|
}
|
|
}
|
|
|
|
} // namespace logic
|