@@ -33,10 +33,12 @@ USB_INCLUDES = -Imiddlewares/ST/STM32_US
USB_INCLUDES += -Imiddlewares/ST/STM32_USB_Device_Library/Class/CDC/Inc
# USER_CFLAGS: user C flags (enable warnings, enable debug info)
USER_CFLAGS = -Wall -g -ffunction-sections -fno-exceptions -fdata-sections -Os
USER_CFLAGS = -Wall -g -ffunction-sections -fno-exceptions -fdata-sections -Os --function-sections -fno-common
# USER_LDFLAGS: user LD flags
USER_LDFLAGS = -fno-exceptions -ffunction-sections -fno-exceptions -fdata-sections -Wl,--gc-sections
USER_LDFLAGS += --static
# TARGET_DEVICE: device to compile for
TARGET_DEVICE = STM32F042x6
@@ -2,8 +2,8 @@
#define CONFIG_H
// Temperature sensor type
//#define MAX31855_TC_SENSOR
#define MAX31865_RTD_SENSOR
#define MAX31855_TC_SENSOR
//#define MAX31865_RTD_SENSOR
// Virtual serial port transmit rate
@@ -73,7 +73,10 @@ void display_process(therm_settings_t* s
case MODE_HEAT:
{
ssd1306_drawstring("-> heat ", 1, 40);
if(set->val.plant_type == PLANT_HEATER)
else
ssd1306_drawstring("-> cool ", 1, 40);
} break;
case MODE_SETUP:
@@ -101,7 +104,7 @@ void display_process(therm_settings_t* s
status->state = STATE_PREHEAT;
break;
status->state = STATE_SETP;
status->state = STATE_SETMODE;
case MODE_RESET:
status->state = STATE_RESET;
@@ -134,6 +137,73 @@ void display_process(therm_settings_t* s
case STATE_SETMODE:
// Write text to OLED
// [ therm :: set mode ]
// [ m = ]
ssd1306_drawstring("Control Mode", 0, 40);
ssd1306_drawlogo();
if(set->val.control_mode == MODE_PID)
ssd1306_drawstring("PID ", 1, 60);
ssd1306_drawstring("Thermostat", 1, 60);
ssd1306_drawstring("Press to accept", 3, 40);
// Button handler
if(SW_BTN_PRESSED) {
status->state = STATE_SETPLANTTYPE;
}
else if (!HAL_GPIO_ReadPin(SW_UP)) {
set->val.control_mode = MODE_PID;
else if(!HAL_GPIO_ReadPin(SW_DOWN)) {
set->val.control_mode = MODE_THERMOSTAT;
// Event Handler
// N/A
case STATE_SETPLANTTYPE:
ssd1306_drawstring("Plant Type", 0, 40);
ssd1306_drawstring("Heater", 1, 60);
ssd1306_drawstring("Cooler", 1, 60);
status->state = STATE_SETBOOTTOBREW;
set->val.plant_type = PLANT_COOLER;
set->val.plant_type = PLANT_HEATER;
case STATE_SETP:
@@ -378,8 +448,12 @@ void display_process(therm_settings_t* s
// [ therm : ready to brew ]
// [ 30 => 120 C ]
ssd1306_drawstring("Preheated!", 0, 0);
//ssd1306_drawlogo();
ssd1306_drawstring("Precooled!", 0, 0);
draw_setpoint(status);
status->pid_enabled = 1;
status->setpoint = set->val.setpoint_brew;
@@ -89,6 +89,7 @@ int main(void)
uint32_t last_vcp_tx = 0;
uint32_t last_led = 0;
uint32_t last_pid = 0;
uint32_t last_thermostat = 0;
int16_t ssr_output = 0; // Duty cycle of ssr, 0 to SSR_PERIOD
// Main loop
@@ -101,7 +102,7 @@ int main(void)
last_led = HAL_GetTick();
if((HAL_GetTick() - last_pid > PID_PERIOD))
if(set.val.control_mode == MODE_PID && (HAL_GetTick() - last_pid > PID_PERIOD))
#ifdef MAX31865_RTD_SENSOR
@@ -115,6 +116,10 @@ int main(void)
// Get ssr output for next time
int16_t power_percent = pid_update(&set, &status, &pid_state);
if(set.val.plant_type == PLANT_HEATER)
power_percent *= -1;
//power-percent is 0-1000?
ssr_output = power_percent; //(((uint32_t)SSR_PERIOD * (uint32_t)10 * (uint32_t)100) * power_percent) / (uint32_t)1000000;
@@ -134,20 +139,19 @@ int main(void)
// Kill SSR once the desired on-time has elapsed
if(HAL_GetTick() - last_ssr_on > ssr_output || ssr_output <= 0)
if(set.val.control_mode == MODE_PID && (HAL_GetTick() - last_ssr_on > ssr_output || ssr_output <= 0))
HAL_GPIO_WritePin(SSR_PIN, 0);
HAL_GPIO_WritePin(LED_POWER, 0);
// Every 200ms, set the SSR on unless output is 0
if(HAL_GetTick() - last_ssr_on > SSR_PERIOD)
if(set.val.control_mode == MODE_PID && HAL_GetTick() - last_ssr_on > SSR_PERIOD)
// Only support heating (ssr_output > 0) right now
// Heat or cool, if we need to
if(ssr_output > 0)
HAL_GPIO_WritePin(SSR_PIN, 1);
HAL_GPIO_WritePin(LED_POWER, 1);
last_ssr_on = HAL_GetTick();
@@ -160,6 +164,55 @@ int main(void)
// Thermostatic control
if(set.val.control_mode == MODE_THERMOSTAT && HAL_GetTick() - last_thermostat > SSR_PERIOD)
max31865_readtemp(spi_get(), &set, &status);
#else
max31855_readtemp(spi_get(), &set, &status); // Read MAX31855
#endif
// TODO: Migrate this FxP conversion to the readtemp code or similar
int8_t temp_frac = status.temp_frac > 9 ? status.temp_frac / 10 : status.temp_frac;
temp_frac = status.temp > 0 ? temp_frac : temp_frac * -1;
int32_t temp = (status.temp * 10) + temp_frac;
uint8_t plant_on = 0;
// EMZ FIXME: This could be way simpler
if(set.val.plant_type == PLANT_HEATER && status.setpoint * 10 < temp)
plant_on = 1;
else if(set.val.plant_type == PLANT_COOLER && status.setpoint * 10 > temp)
// EMZ: TODO: Refactor to output_enabled or something
if(status.pid_enabled && plant_on)
// EMZ TODO: functionalize this
// put ssr output on display
ssd1306_drawstring(" ", 0, 90); //fixme: this is bad, but I can't get the old digits to clear otherwise
char tempstr[6];
itoa(ssr_output, tempstr, 10);
ssd1306_drawstring(tempstr, 0, 90);
last_thermostat = HAL_GetTick();
// Transmit temperature over USB-CDC on a regular basis
if(HAL_GetTick() - last_vcp_tx > VCP_TX_FREQ)
@@ -27,6 +27,8 @@ typedef union
uint32_t ignore_error;
int32_t setpoint_brew;
int32_t setpoint_steam;
uint32_t control_mode;
uint32_t plant_type;
} val;
uint16_t data[128];
@@ -39,7 +41,9 @@ enum tempunits {
enum state {
STATE_IDLE = 0,
STATE_SETMODE,
STATE_SETPLANTTYPE,
STATE_SETP,
STATE_SETI,
STATE_SETD,
@@ -56,6 +60,16 @@ enum state {
STATE_RESET,
};
enum control_mode {
MODE_PID = 0,
MODE_THERMOSTAT,
enum plant_type {
PLANT_HEATER = 0,
PLANT_COOLER,
enum GOTO_MODE {
#ifdef BOOTLOADER_SHORTCUT
MODE_BOOTLOADER,
Status change: