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
468 lines
17 KiB
C++
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);
|
|
}
|
|
}
|