
Implemented F, H and R parameters to M106 command. The second fan output on a Duet 0.8.5 now defaults to being a thermostatic fan at power up. Improved speed of file upload to SD card G32 is now allowed if the printer has not been homed, if there is a bed.g file G30 commands are no longer allowed on a delta that has not been homed M572 parameter P (drive number) replaced by parameter D (extruder number) File info requests are now processed in stages to reduce impact on printing (thanks chrishamm) Use latest network stack and webserver modules from chrishamm (thanks chrishamm) Added Roland mill support (thanks RRP/chrishamm) Added S parameter (idle timeout) to M18 ans M84 commands (thanks chrishamm) Moved I/O pin assignments to separate Pins.h file to more easily support alternative hardware (thanks dnewman) Bug fix: filament usage and % print complete figures were incorrect when absolute extruder coordinates were used Bug fix: file-based print estimate was occasionally returned as 'inf' which caused the web interface to disconnect Bug fix: M666 now flags all towers as not homed Bug fixes to extruder pressure compensation (M572 command).
335 lines
7.4 KiB
C++
335 lines
7.4 KiB
C++
/****************************************************************************************************
|
|
|
|
RepRapFirmware - Tool
|
|
|
|
This class implements a tool in the RepRap machine, usually (though not necessarily) an extruder.
|
|
|
|
Tools may have zero or more drives associated with them and zero or more heaters. There are a fixed number
|
|
of tools in a given RepRap, with fixed heaters and drives. All this is specified on reboot, and cannot
|
|
be altered dynamically. This restriction may be lifted in the future. Tool descriptions are stored in
|
|
GCode macros that are loaded on reboot.
|
|
|
|
-----------------------------------------------------------------------------------------------------
|
|
|
|
Version 0.1
|
|
|
|
Created on: Apr 11, 2014
|
|
|
|
Adrian Bowyer
|
|
RepRap Professional Ltd
|
|
http://reprappro.com
|
|
|
|
Licence: GPL
|
|
|
|
****************************************************************************************************/
|
|
|
|
#include "RepRapFirmware.h"
|
|
|
|
Tool * Tool::freelist = nullptr;
|
|
|
|
/*static*/ Tool * Tool::Create(int toolNumber, long d[], size_t dCount, long h[], size_t hCount)
|
|
{
|
|
if (dCount > DRIVES - AXES)
|
|
{
|
|
reprap.GetPlatform()->Message(GENERIC_MESSAGE, "Error: Tool creation: attempt to use more drives than there are in the RepRap");
|
|
return nullptr;
|
|
}
|
|
|
|
if (hCount > HEATERS)
|
|
{
|
|
reprap.GetPlatform()->Message(GENERIC_MESSAGE, "Error: Tool creation: attempt to use more heaters than there are in the RepRap");
|
|
return nullptr;
|
|
}
|
|
|
|
Tool *t;
|
|
if (freelist != nullptr)
|
|
{
|
|
t = freelist;
|
|
freelist = t->next;
|
|
}
|
|
else
|
|
{
|
|
t = new Tool;
|
|
}
|
|
|
|
t->myNumber = toolNumber;
|
|
t->next = nullptr;
|
|
t->active = false;
|
|
t->driveCount = dCount;
|
|
t->heaterCount = hCount;
|
|
t->heaterFault = false;
|
|
t->mixing = false;
|
|
t->displayColdExtrudeWarning = false;
|
|
|
|
for(size_t axis = 0; axis < AXES; axis++)
|
|
{
|
|
t->offset[axis] = 0.0;
|
|
}
|
|
|
|
if (t->driveCount > 0)
|
|
{
|
|
float r = 1.0/(float)(t->driveCount);
|
|
|
|
for (size_t drive = 0; drive < t->driveCount; drive++)
|
|
{
|
|
t->drives[drive] = d[drive];
|
|
t->mix[drive] = r;
|
|
}
|
|
}
|
|
|
|
if (t->heaterCount > 0)
|
|
{
|
|
for(size_t heater = 0; heater < t->heaterCount; heater++)
|
|
{
|
|
t->heaters[heater] = h[heater];
|
|
t->activeTemperatures[heater] = ABS_ZERO;
|
|
t->standbyTemperatures[heater] = ABS_ZERO;
|
|
}
|
|
}
|
|
|
|
return t;
|
|
}
|
|
|
|
/*static*/ void Tool::Delete(Tool *t)
|
|
{
|
|
if (t != nullptr)
|
|
{
|
|
t->next = freelist;
|
|
freelist = t;
|
|
}
|
|
}
|
|
|
|
void Tool::Print(StringRef& reply)
|
|
{
|
|
reply.printf("Tool %d - drives: ", myNumber);
|
|
char comma = ',';
|
|
for (size_t drive = 0; drive < driveCount; drive++)
|
|
{
|
|
if (drive >= driveCount - 1)
|
|
{
|
|
comma = ';';
|
|
}
|
|
reply.catf("%d%c", drives[drive], comma);
|
|
}
|
|
|
|
reply.cat(" heaters (active/standby temps): ");
|
|
comma = ',';
|
|
for (size_t heater = 0; heater < heaterCount; heater++)
|
|
{
|
|
if (heater >= heaterCount - 1)
|
|
{
|
|
comma = ';';
|
|
}
|
|
reply.catf("%d (%.1f/%.1f)%c", heaters[heater],
|
|
activeTemperatures[heater], standbyTemperatures[heater], comma);
|
|
}
|
|
|
|
reply.catf(" status: %s", active ? "selected" : "standby");
|
|
}
|
|
|
|
float Tool::MaxFeedrate() const
|
|
{
|
|
if (driveCount <= 0)
|
|
{
|
|
reprap.GetPlatform()->Message(GENERIC_MESSAGE, "Error: Attempt to get maximum feedrate for a tool with no drives.\n");
|
|
return 1.0;
|
|
}
|
|
float result = 0.0;
|
|
for (size_t d = 0; d < driveCount; d++)
|
|
{
|
|
float mf = reprap.GetPlatform()->MaxFeedrate(drives[d] + AXES);
|
|
if (mf > result)
|
|
{
|
|
result = mf;
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
float Tool::InstantDv() const
|
|
{
|
|
if (driveCount <= 0)
|
|
{
|
|
reprap.GetPlatform()->Message(GENERIC_MESSAGE, "Error: Attempt to get InstantDv for a tool with no drives.\n");
|
|
return 1.0;
|
|
}
|
|
float result = FLT_MAX;
|
|
for (size_t d = 0; d < driveCount; d++)
|
|
{
|
|
float idv = reprap.GetPlatform()->ActualInstantDv(drives[d] + AXES);
|
|
if (idv < result)
|
|
{
|
|
result = idv;
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
// There is a temperature fault on a heater.
|
|
// Disable all tools using that heater.
|
|
// This function must be called for the first
|
|
// entry in the linked list.
|
|
|
|
void Tool::FlagTemperatureFault(int8_t heater)
|
|
{
|
|
Tool* n = this;
|
|
while(n != nullptr)
|
|
{
|
|
n->SetTemperatureFault(heater);
|
|
n = n->Next();
|
|
}
|
|
}
|
|
|
|
void Tool::ClearTemperatureFault(int8_t heater)
|
|
{
|
|
Tool* n = this;
|
|
while(n != nullptr)
|
|
{
|
|
n->ResetTemperatureFault(heater);
|
|
n = n->Next();
|
|
}
|
|
}
|
|
|
|
void Tool::SetTemperatureFault(int8_t dudHeater)
|
|
{
|
|
for (size_t heater = 0; heater < heaterCount; heater++)
|
|
{
|
|
if(dudHeater == heaters[heater])
|
|
{
|
|
heaterFault = true;
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
void Tool::ResetTemperatureFault(int8_t wasDudHeater)
|
|
{
|
|
for (size_t heater = 0; heater < heaterCount; heater++)
|
|
{
|
|
if (wasDudHeater == heaters[heater])
|
|
{
|
|
heaterFault = false;
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
bool Tool::AllHeatersAtHighTemperature(bool forExtrusion) const
|
|
{
|
|
for (size_t heater = 0; heater < heaterCount; heater++)
|
|
{
|
|
const float temperature = reprap.GetHeat()->GetTemperature(heaters[heater]);
|
|
if (temperature < HOT_ENOUGH_TO_RETRACT || (temperature < HOT_ENOUGH_TO_EXTRUDE && forExtrusion))
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void Tool::Activate(Tool* currentlyActive)
|
|
{
|
|
if (!active)
|
|
{
|
|
if (currentlyActive != nullptr && currentlyActive != this)
|
|
{
|
|
currentlyActive->Standby();
|
|
}
|
|
for (size_t heater = 0; heater < heaterCount; heater++)
|
|
{
|
|
reprap.GetHeat()->SetActiveTemperature(heaters[heater], activeTemperatures[heater]);
|
|
reprap.GetHeat()->SetStandbyTemperature(heaters[heater], standbyTemperatures[heater]);
|
|
reprap.GetHeat()->Activate(heaters[heater]);
|
|
}
|
|
active = true;
|
|
}
|
|
}
|
|
|
|
void Tool::Standby()
|
|
{
|
|
if (active)
|
|
{
|
|
for (size_t heater = 0; heater < heaterCount; heater++)
|
|
{
|
|
reprap.GetHeat()->SetStandbyTemperature(heaters[heater], standbyTemperatures[heater]);
|
|
reprap.GetHeat()->Standby(heaters[heater]);
|
|
}
|
|
active = false;
|
|
}
|
|
}
|
|
|
|
void Tool::SetVariables(const float* standby, const float* active)
|
|
{
|
|
for (size_t heater = 0; heater < heaterCount; heater++)
|
|
{
|
|
if (active[heater] < NEARLY_ABS_ZERO && standby[heater] < NEARLY_ABS_ZERO)
|
|
{
|
|
// Temperatures close to ABS_ZERO turn off all associated heaters
|
|
reprap.GetHeat()->SwitchOff(heaters[heater]);
|
|
}
|
|
else
|
|
{
|
|
if (active[heater] < BAD_HIGH_TEMPERATURE)
|
|
{
|
|
activeTemperatures[heater] = active[heater];
|
|
reprap.GetHeat()->SetActiveTemperature(heaters[heater], activeTemperatures[heater]);
|
|
}
|
|
if (standby[heater] < BAD_HIGH_TEMPERATURE)
|
|
{
|
|
standbyTemperatures[heater] = standby[heater];
|
|
reprap.GetHeat()->SetStandbyTemperature(heaters[heater], standbyTemperatures[heater]);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void Tool::GetVariables(float* standby, float* active) const
|
|
{
|
|
for (size_t heater = 0; heater < heaterCount; heater++)
|
|
{
|
|
active[heater] = activeTemperatures[heater];
|
|
standby[heater] = standbyTemperatures[heater];
|
|
}
|
|
}
|
|
|
|
// May be called from ISR
|
|
bool Tool::ToolCanDrive(bool extrude)
|
|
{
|
|
if (heaterFault)
|
|
return false;
|
|
|
|
if (reprap.GetHeat()->ColdExtrude() || AllHeatersAtHighTemperature(extrude))
|
|
return true;
|
|
|
|
displayColdExtrudeWarning = true;
|
|
return false;
|
|
}
|
|
|
|
// Update the number of active drives and extruders in use to reflect what this tool uses
|
|
void Tool::UpdateExtruderAndHeaterCount(uint16_t &numExtruders, uint16_t &numHeaters) const
|
|
{
|
|
for (size_t drive = 0; drive < driveCount; drive++)
|
|
{
|
|
if (drives[drive] >= numExtruders)
|
|
{
|
|
numExtruders = drives[drive] + 1;
|
|
}
|
|
}
|
|
|
|
for (size_t heater = 0; heater < heaterCount; heater++)
|
|
{
|
|
if (heaters[heater] != BED_HEATER && heaters[heater] >= numHeaters)
|
|
{
|
|
numHeaters = heaters[heater] + 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
bool Tool::DisplayColdExtrudeWarning()
|
|
{
|
|
bool result = displayColdExtrudeWarning;
|
|
displayColdExtrudeWarning = false;
|
|
return result;
|
|
}
|
|
|
|
// End
|