cable-length-cutter/main/guide-stepper.cpp
jonny_l480 4112d8653c Add dynamic-winding-length depending on targetLength
When winding short lengths e.g. 5m it does not make sense
to use the entire winding width thus getting unnecessary
wide cable ring.

Determined some thresholds while testing and implemented
those to be applied at target length change automatically:

control:
    - update winding width each time target length changes

guide-stepper:
    - add function that returns dynamic winding width depending
      on passed target length according to fixed thresholds
2024-03-17 09:41:24 +01:00

468 lines
17 KiB
C++

extern "C"
{
#include <stdio.h>
#include "freertos/FreeRTOS.h"
#include "freertos/task.h"
#include "esp_log.h"
#include "driver/adc.h"
#include "freertos/queue.h"
#include "freertos/semphr.h"
}
#include "stepper.hpp"
#include "config.h"
#include "global.hpp"
#include "guide-stepper.hpp"
#include "encoder.hpp"
#include "shutdown.hpp"
//macro to get smaller value out of two
#define MIN(a, b) (((a) < (b)) ? (a) : (b))
//---------------------
//--- configuration ---
//---------------------
//also see config.h
//for pin definitions and guide parameters
// configure testing modes
#define STEPPER_TEST_TRAVEL 65 // mm
// speeds for testing with potentiometer (test task only)
#define SPEED_MIN 2.0 // mm/s
#define SPEED_MAX 70.0 // mm/s
//note: actual speed is currently defined in config.h with STEPPER_SPEED_DEFAULT
//simulate encoder with reset button to test stepper ctl task
//note STEPPER_TEST has to be defined as well
//#define STEPPER_SIMULATE_ENCODER
#define PI 3.14159
//#define POS_MAX_STEPS GUIDE_MAX_MM * STEPPER_STEPS_PER_MM //note replaced with posMaxSteps
#define POS_MIN_STEPS GUIDE_MIN_MM * STEPPER_STEPS_PER_MM
//----------------------
//----- variables ------
//----------------------
typedef enum axisDirection_t {AXIS_MOVING_LEFT = 0, AXIS_MOVING_RIGHT} axisDirection_t;
static const char *TAG = "stepper-ctrl"; //tag for logging
static axisDirection_t currentAxisDirection = AXIS_MOVING_RIGHT;
static uint32_t posNow = 0;
static int layerCount = 0;
// queue for sending commands to task handling guide movement
static QueueHandle_t queue_commandsGuideTask;
// mutex to prevent multiple axis to config variables also accessed/modified by control task
SemaphoreHandle_t configVariables_mutex = xSemaphoreCreateMutex();
// configured winding width: position the axis returns again in steps
static uint32_t posMaxSteps = GUIDE_MAX_MM * STEPPER_STEPS_PER_MM; //assign default width
//----------------------
//----- functions ------
//----------------------
//=============================
//=== guide_getAxisPosSteps ===
//=============================
// return local variable posNow
// needed at shutdown detection to store last axis position in nvs
int guide_getAxisPosSteps(){
return posNow;
}
//=============================
//=== guide_setWindingWidth ===
//=============================
// set custom winding width (axis position the guide returns in mm)
void guide_setWindingWidth(uint8_t maxPosMm)
{
if (xSemaphoreTake(configVariables_mutex, portMAX_DELAY) == pdTRUE) // mutex to prevent multiple access by control and stepper-ctl task
{
posMaxSteps = maxPosMm * STEPPER_STEPS_PER_MM;
ESP_LOGI(TAG, "set winding width / max pos to %dmm", maxPosMm);
xSemaphoreGive(configVariables_mutex);
}
}
//=======================================
//=== guide_targetLength2WindingWidth ===
//=======================================
// calculate dynamic winding width in mm from cable length in mm
uint8_t guide_targetLength2WindingWidth(int lenMm)
{
#ifdef DYNAMIC_WINDING_WIDTH_ENABLED
uint8_t width;
//--- config ---
// define thresholds for winding widths according to target length:
if (lenMm <= 5000) // 0-5m
width = 15;
else if (lenMm <= 10000) // 6-10m
width = 25;
else if (lenMm <= 15000) // 11-15m
width = 30;
else if (lenMm <= 25000) // 16-25m
width = 65;
else // >25m
width = GUIDE_MAX_MM;
ESP_LOGW(TAG, "length2width: calculated windingWidth=%dmm from targetLength=%dm", width, lenMm);
return width;
#else
ESP_LOGD(TAG, "length2width: dynamic windingWidh not enabled, stay at GUIDE_MAX=%d", GUIDE_MAX_MM);
return GUIDE_MAX_MM;
#endif
//TODO update winding width here as well already?
}
//=============================
//=== guide_getWindingWidth ===
//=============================
// get currently configured winding width (axis position at which the guide returns in mm)
uint8_t guide_getWindingWidth()
{
if (xSemaphoreTake(configVariables_mutex, portMAX_DELAY) == pdTRUE) // mutex to prevent multiple access by control and stepper-ctl task
{
uint8_t returnValue = posMaxSteps / STEPPER_STEPS_PER_MM;
xSemaphoreGive(configVariables_mutex);
return returnValue;
}
return 0;
}
//==========================
//==== guide_moveToZero ====
//==========================
//tell stepper-control task to move cable guide to zero position
void guide_moveToZero(){
bool valueToSend = true; // or false
xQueueSend(queue_commandsGuideTask, &valueToSend, portMAX_DELAY);
ESP_LOGI(TAG, "sending command to stepper_ctl task via queue");
}
//---------------------
//---- travelSteps ----
//---------------------
//move axis certain Steps (relative) between left and right or reverse when negative
void travelSteps(int stepsTarget){
//TODO simplify this function, one simple calculation of new position?
//with new custom driver no need to detect direction change
// cancel when width is zero or no steps received
if (posMaxSteps == 0 || stepsTarget == 0){
ESP_LOGD(TAG, "travelSteps: MaxSteps or stepsTarget = 0 -> nothing to do");
return;
}
int stepsToGo, remaining;
stepsToGo = abs(stepsTarget);
// invert direction in reverse mode (cable gets spooled off reel)
if (stepsTarget < 0) {
currentAxisDirection = (currentAxisDirection == AXIS_MOVING_LEFT) ? AXIS_MOVING_RIGHT : AXIS_MOVING_LEFT; //toggle between RIGHT<->Left
}
while (stepsToGo != 0){
//--- currently moving right ---
if (currentAxisDirection == AXIS_MOVING_RIGHT){ //currently moving right
if (xSemaphoreTake(configVariables_mutex, portMAX_DELAY) == pdTRUE) { //prevent multiple acces on posMaxSteps by control-task
remaining = posMaxSteps - posNow; //calc remaining distance fom current position to limit
if (stepsToGo > remaining){ //new distance will exceed limit
stepper_setTargetPosSteps(posMaxSteps); //move to limit
stepper_waitForStop(1000);
posNow = posMaxSteps;
currentAxisDirection = AXIS_MOVING_LEFT; //change current direction for next iteration
//increment/decrement layer count depending on current cable direction
layerCount += (stepsTarget > 0) - (stepsTarget < 0);
if (layerCount < 0) layerCount = 0; //negative layers are not possible
stepsToGo = stepsToGo - remaining; //decrease target length by already traveled distance
ESP_LOGI(TAG, " --- moved to max -> change direction (L) --- \n ");
}
else { //target distance does not reach the limit
stepper_setTargetPosSteps(posNow + stepsToGo); //move by (remaining) distance to reach target length
ESP_LOGD(TAG, "moving to %d\n", posNow+stepsToGo);
posNow += stepsToGo;
stepsToGo = 0; //finished, reset target length (could as well exit loop/break)
}
xSemaphoreGive(configVariables_mutex);
}
}
//--- currently moving left ---
else if (currentAxisDirection == AXIS_MOVING_LEFT){
remaining = posNow - POS_MIN_STEPS;
if (stepsToGo > remaining){
stepper_setTargetPosSteps(POS_MIN_STEPS);
stepper_waitForStop(1000);
posNow = POS_MIN_STEPS;
currentAxisDirection = AXIS_MOVING_RIGHT; //switch direction
//increment/decrement layer count depending on current cable direction
layerCount += (stepsTarget > 0) - (stepsTarget < 0);
if (layerCount < 0) layerCount = 0; //negative layers are not possible
stepsToGo = stepsToGo - remaining;
ESP_LOGI(TAG, " --- moved to min -> change direction (R) --- \n ");
}
else {
stepper_setTargetPosSteps(posNow - stepsToGo); //when moving left the coordinate has to be decreased
ESP_LOGD(TAG, "moving to %d\n", posNow - stepsToGo);
posNow -= stepsToGo;
stepsToGo = 0;
}
}
}
// undo inversion of currentAxisDirection after reverse mode is finished
if (stepsTarget < 0) {
currentAxisDirection = (currentAxisDirection == AXIS_MOVING_LEFT) ? AXIS_MOVING_RIGHT : AXIS_MOVING_LEFT; //toggle between RIGHT<->Left
}
return;
}
//------------------
//---- travelMm ----
//------------------
//move axis certain Mm (relative) between left and right or reverse when negative
void travelMm(int length){
travelSteps(length * STEPPER_STEPS_PER_MM);
}
//----------------------
//---- init_stepper ----
//----------------------
//initialize/configure stepper instance
void init_stepper() {
//TODO unnecessary wrapper?
ESP_LOGW(TAG, "initializing stepper...");
stepper_init();
// create queue for sending commands to task handling guide movement
// currently length 1 and only one command possible, thus bool
queue_commandsGuideTask = xQueueCreate(1, sizeof(bool));
}
//--------------------------
//--- updateSpeedFromAdc ---
//--------------------------
//function that updates speed value using ADC input and configured MIN/MAX - used for testing only
void updateSpeedFromAdc() {
int potiRead = gpio_readAdc(ADC_CHANNEL_POTI); //0-4095 GPIO34
double poti = potiRead/4095.0;
int speed = poti*(SPEED_MAX-SPEED_MIN) + SPEED_MIN;
stepper_setSpeed(speed);
ESP_LOGW(TAG, "poti: %d (%.2lf%%), set speed to: %d", potiRead, poti*100, speed);
}
//============================
//==== TASK stepper_test =====
//============================
//test axis without using encoder input
#ifndef STEPPER_SIMULATE_ENCODER
void task_stepper_test(void *pvParameter)
{
stepper_init();
while(1){
vTaskDelay(20 / portTICK_PERIOD_MS);
//------ handle switches ------
//run handle functions for all switches
SW_START.handle();
SW_RESET.handle();
SW_SET.handle();
SW_PRESET1.handle();
SW_PRESET2.handle();
SW_PRESET3.handle();
SW_CUT.handle();
SW_AUTO_CUT.handle();
#ifdef ONE_BUTTON_TEST //test with "reset-button" only
static int state = 0;
//cycle through test commands with one button
if (SW_RESET.risingEdge) {
switch (state){
case 0:
stepper_setTargetPosMm(50);
//stepper_setTargetPosSteps(1000);
state++;
break;
case 1:
stepper_setTargetPosMm(80);
//stepper_setTargetPosSteps(100);
state++;
break;
case 2:
stepper_setTargetPosMm(20);
//stepper_setTargetPosSteps(100);
state++;
break;
case 3:
stepper_setTargetPosMm(60);
//stepper_setTargetPosSteps(2000);
state = 0;
break;
}
}
#else //test with all buttons
if (SW_RESET.risingEdge) {
buzzer.beep(1, 500, 100);
stepper_setTargetPosMm(0);
}
if (SW_PRESET1.risingEdge) {
buzzer.beep(1, 200, 100);
stepper_setTargetPosMm(50);
}
if (SW_PRESET2.risingEdge) {
buzzer.beep(2, 200, 100);
stepper_setTargetPosMm(75);
}
if (SW_PRESET3.risingEdge) {
buzzer.beep(3, 200, 100);
stepper_setTargetPosMm(100);
}
#endif
}
}
#endif //end SIMULATE_ENCODER
//============================
//===== TASK stepper_ctl =====
//============================
//task controlling the linear axis guiding the cable according to wire length spooled
#ifdef STEPPER_SIMULATE_ENCODER
void task_stepper_test(void *pvParameter)
#else
void task_stepper_ctl(void *pvParameter)
#endif
{
//-- variables --
static int encStepsPrev = 0; //measured encoder steps at last run
static double travelStepsPartial = 0; //store resulted remaining partial steps last run
//temporary variables for calculating required steps, or logging
int encStepsNow = 0; //get curretly measured steps of encoder
int encStepsDelta = 0; //encoder steps changed since last iteration
double cableLen = 0;
double travelStepsExact = 0; //steps axis has to travel
int travelStepsFull = 0;
double travelMm = 0;
double turns = 0;
float currentDiameter;
//initialize stepper at task start
init_stepper();
//define zero-position
// use last known position stored at last shutdown to reduce time crashing into hardware limit
int posLastShutdown = nvsReadLastAxisPosSteps();
if (posLastShutdown >= 0)
{ // would be -1 when failed
ESP_LOGW(TAG, "auto-home: considerting pos last shutdown %dmm + tolerance %dmm",
posLastShutdown / STEPPER_STEPS_PER_MM,
AUTO_HOME_TRAVEL_ADD_TO_LAST_POS_MM);
// home considering last position and offset, but limit to max distance possible
stepper_home(MIN((posLastShutdown/STEPPER_STEPS_PER_MM + AUTO_HOME_TRAVEL_ADD_TO_LAST_POS_MM), MAX_TOTAL_AXIS_TRAVEL_MM));
}
else { // default to max travel when read from nvs failed
stepper_home(MAX_TOTAL_AXIS_TRAVEL_MM);
}
//repeatedly read changes in measured cable length and move axis accordingly
while(1){
// guide is disabled when posMaxSteps is zero:
if (posMaxSteps == 0)
{
// move to starting position
stepper_setTargetPosSteps(0);
vTaskDelay(1000 / portTICK_PERIOD_MS);
ESP_LOGD(TAG, "posMaxSteps is zero -> guide disabled, doing nothing");
// stop loop iteration
continue;
}
#ifdef STEPPER_SIMULATE_ENCODER
//TESTING - simulate encoder using switch
SW_RESET.handle();
//note
if (SW_RESET.risingEdge) encStepsNow += 500;
#else
//get current length
encStepsNow = encoder_getSteps();
#endif
// move to zero and reset if command received via queue
bool receivedValue;
if (xQueueReceive(queue_commandsGuideTask, &receivedValue, 0) == pdTRUE)
{
// Process the received value
// TODO support other commands (currently only move to zero possible)
ESP_LOGW(TAG, "task: move-to-zero command received via queue, starting move, waiting until position reached");
stepper_setTargetPosMm(0);
stepper_waitForStop();
//reset variables -> start tracking cable movement starting from position zero
// ensure stepsDelta is 0
encStepsNow = encoder_getSteps();
encStepsPrev = encStepsNow;
travelStepsPartial = 0;
// set locally stored axis position and counted layers to 0 (used for calculating the target axis coordinate and steps)
posNow = 0;
layerCount = 0;
currentAxisDirection = AXIS_MOVING_RIGHT;
ESP_LOGW(TAG, "at position 0, reset variables, resuming normal cable guiding operation");
}
//calculate change
encStepsDelta = encStepsNow - encStepsPrev;
//check if reset happend without moving to zero before - resulting in huge diff
if (encStepsDelta != 0 && encStepsNow == 0){ // this should not happen and causes weird movement
ESP_LOGE(TAG, "encoder steps changed to 0 (reset) without previous moveToZero() call, resulting in stepsDelta=%d", encStepsDelta);
}
//read potentiometer and normalize (0-1) to get a variable for testing
//float potiModifier = (float) gpio_readAdc(ADC_CHANNEL_POTI) / 4095; //0-4095 -> 0-1
//ESP_LOGI(TAG, "current poti-modifier = %f", potiModifier);
//calculate steps to move
cableLen = (double)encStepsDelta * 1000 / ENCODER_STEPS_PER_METER;
//effective diameter increases each layer
currentDiameter = D_REEL + LAYER_THICKNESS_MM * 2 * layerCount;
turns = cableLen / (PI * currentDiameter);
travelMm = turns * D_CABLE;
travelStepsExact = travelMm * STEPPER_STEPS_PER_MM + travelStepsPartial; //convert mm to steps and add not moved partial steps
travelStepsPartial = 0;
travelStepsFull = (int)travelStepsExact;
//move axis when ready to move at least 1 full step
if (abs(travelStepsFull) > 1){
travelStepsPartial = fmod(travelStepsExact, 1); //save remaining partial steps to be added in the next iteration
ESP_LOGI(TAG, "dCablelen=%.2lf, dTurns=%.2lf, travelMm=%.3lf, StepsExact: %.3lf, StepsFull=%d, StepsPartial=%.3lf, totalLayerCount=%d, diameter=%.1f", cableLen, turns, travelMm, travelStepsExact, travelStepsFull, travelStepsPartial, layerCount, currentDiameter);
ESP_LOGD(TAG, "MOVING %d steps", travelStepsFull);
travelSteps(travelStepsExact);
encStepsPrev = encStepsNow; //update previous length
}
else {
//TODO use encoder queue to only run this check at encoder event?
vTaskDelay(5);
}
vTaskDelay(5 / portTICK_PERIOD_MS);
}
}