
Changed network startup to avoid long delay if no network cable is connected Enable DHCP support if IP address is configured as 0.0.0.0 (not working yet) Move thermistor parameters to M305, added parameter to configure series resistance Added B and W PID parameters
183 lines
4.1 KiB
C++
183 lines
4.1 KiB
C++
/****************************************************************************************************
|
|
|
|
RepRapFirmware - Heat
|
|
|
|
This is all the code to deal with heat and temperature.
|
|
|
|
-----------------------------------------------------------------------------------------------------
|
|
|
|
Version 0.1
|
|
|
|
18 November 2012
|
|
|
|
Adrian Bowyer
|
|
RepRap Professional Ltd
|
|
http://reprappro.com
|
|
|
|
Licence: GPL
|
|
|
|
****************************************************************************************************/
|
|
|
|
#include "RepRapFirmware.h"
|
|
|
|
Heat::Heat(Platform* p, GCodes* g)
|
|
{
|
|
platform = p;
|
|
gCodes = g;
|
|
for(int8_t heater=0; heater < HEATERS; heater++)
|
|
pids[heater] = new PID(platform, heater);
|
|
active = false;
|
|
}
|
|
|
|
void Heat::Init()
|
|
{
|
|
for(int8_t heater=0; heater < HEATERS; heater++)
|
|
pids[heater]->Init();
|
|
lastTime = platform->Time();
|
|
longWait = lastTime;
|
|
active = true;
|
|
}
|
|
|
|
void Heat::Exit()
|
|
{
|
|
platform->Message(HOST_MESSAGE, "Heat class exited.\n");
|
|
active = false;
|
|
}
|
|
|
|
void Heat::Spin()
|
|
{
|
|
if(!active)
|
|
return;
|
|
|
|
float t = platform->Time();
|
|
if(t - lastTime < platform->HeatSampleTime())
|
|
return;
|
|
lastTime = t;
|
|
for(int8_t heater=0; heater < HEATERS; heater++)
|
|
pids[heater]->Spin();
|
|
platform->ClassReport("Heat", longWait);
|
|
}
|
|
|
|
void Heat::Diagnostics()
|
|
{
|
|
platform->Message(HOST_MESSAGE, "Heat Diagnostics:\n");
|
|
}
|
|
|
|
bool Heat::AllHeatersAtSetTemperatures()
|
|
{
|
|
float dt;
|
|
for(int8_t heater = 0; heater < HEATERS; heater++)
|
|
{
|
|
dt = GetTemperature(heater);
|
|
if(pids[heater]->Active())
|
|
{
|
|
if(GetActiveTemperature(heater) < TEMPERATURE_LOW_SO_DONT_CARE)
|
|
dt = 0.0;
|
|
else
|
|
dt = fabs(dt - GetActiveTemperature(heater));
|
|
} else
|
|
{
|
|
if(GetStandbyTemperature(heater) < TEMPERATURE_LOW_SO_DONT_CARE)
|
|
dt = 0.0;
|
|
else
|
|
dt = fabs(dt - GetStandbyTemperature(heater));
|
|
}
|
|
if(dt > TEMPERATURE_CLOSE_ENOUGH)
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
//******************************************************************************************************
|
|
|
|
PID::PID(Platform* p, int8_t h)
|
|
{
|
|
platform = p;
|
|
heater = h;
|
|
}
|
|
|
|
void PID::Init()
|
|
{
|
|
platform->SetHeater(heater, 0.0);
|
|
temperature = platform->GetTemperature(heater);
|
|
activeTemperature = ABS_ZERO;
|
|
standbyTemperature = ABS_ZERO;
|
|
lastTemperature = temperature;
|
|
temp_iState = 0.0;
|
|
badTemperatureCount = 0;
|
|
temperatureFault = false;
|
|
active = false;
|
|
}
|
|
|
|
|
|
void PID::Spin()
|
|
{
|
|
if(temperatureFault)
|
|
{
|
|
platform->SetHeater(heater, 0.0); // Make sure...
|
|
return;
|
|
}
|
|
|
|
temperature = platform->GetTemperature(heater);
|
|
|
|
if(temperature < BAD_LOW_TEMPERATURE || temperature > BAD_HIGH_TEMPERATURE)
|
|
{
|
|
badTemperatureCount++;
|
|
if(badTemperatureCount > MAX_BAD_TEMPERATURE_COUNT)
|
|
{
|
|
platform->SetHeater(heater, 0.0);
|
|
temperatureFault = true;
|
|
snprintf(scratchString, STRING_LENGTH, "Temperature measurement fault on heater %d, T = %.1f\n", heater, temperature);
|
|
platform->Message(HOST_MESSAGE, scratchString);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
badTemperatureCount = 0;
|
|
}
|
|
|
|
float error = ((active) ? activeTemperature : standbyTemperature) - temperature;
|
|
const PidParameters& pp = platform->GetPidParameters(heater);
|
|
|
|
if(!pp.UsePID())
|
|
{
|
|
platform->SetHeater(heater, (error > 0.0) ? 1.0 : 0.0);
|
|
return;
|
|
}
|
|
|
|
if(error < -pp.fullBand)
|
|
{
|
|
temp_iState = 0.0;
|
|
platform->SetHeater(heater, 0.0);
|
|
lastTemperature = temperature;
|
|
return;
|
|
}
|
|
if(error > pp.fullBand)
|
|
{
|
|
temp_iState = 0.0;
|
|
platform->SetHeater(heater, 1.0);
|
|
lastTemperature = temperature;
|
|
return;
|
|
}
|
|
|
|
temp_iState += error * pp.kI;
|
|
|
|
if (temp_iState < pp.pidMin) temp_iState = pp.pidMin;
|
|
else if (temp_iState > pp.pidMax) temp_iState = pp.pidMax;
|
|
|
|
float temp_dState = pp.kD * (temperature - lastTemperature);
|
|
float result = pp.kP * error + temp_iState - temp_dState;
|
|
|
|
lastTemperature = temperature;
|
|
|
|
if (result < 0.0) result = 0.0;
|
|
else if (result > 255.0) result = 255.0;
|
|
result = result/255.0;
|
|
|
|
if(!temperatureFault)
|
|
{
|
|
platform->SetHeater(heater, result);
|
|
}
|
|
|
|
//debugPrintf("Heat: e=%f, P=%f, I=%f, d=%f, r=%f\n", error, platform->PidKp(heater)*error, temp_iState, temp_dState, result);
|
|
}
|