From de42b6252ebec3558713cd39334548041654545f Mon Sep 17 00:00:00 2001 From: jonny_ji7 Date: Sun, 12 Mar 2023 13:06:30 +0100 Subject: [PATCH 1/8] Stepper: Add resume/update functionality (WIP) add way to extend current move operation without having to wait for IDLE state (full decel time) --- components/DendoStepper/DendoStepper.cpp | 92 ++++++++++++++++--- .../DendoStepper/include/DendoStepper.h | 8 +- main/guide-stepper.cpp | 27 +++++- 3 files changed, 107 insertions(+), 20 deletions(-) diff --git a/components/DendoStepper/DendoStepper.cpp b/components/DendoStepper/DendoStepper.cpp index 6443845..07d31aa 100644 --- a/components/DendoStepper/DendoStepper.cpp +++ b/components/DendoStepper/DendoStepper.cpp @@ -116,21 +116,35 @@ timer_avail: esp_err_t DendoStepper::runPos(int32_t relative) { - if (!relative) // why would u call it with 0 wtf - return ESP_ERR_NOT_SUPPORTED; - if (ctrl.status > IDLE) - { // we are running, we need to adjust steps accordingly, for now just stop the movement - STEP_LOGW("DendoStepper", "Finising previous move, this command will be ignored"); - return ESP_ERR_NOT_SUPPORTED; - } - + //TODO only enable when actually moving if (ctrl.status == DISABLED) // if motor is disabled, enable it enableMotor(); - ctrl.status = ACC; - setDir(relative < 0); // set CCW if <0, else set CW - currentPos += relative; - calc(abs(relative)); // calculate velocity profile + + if (!relative) // why would u call it with 0 wtf + return ESP_ERR_NOT_SUPPORTED; + + if (ctrl.status > IDLE) { //currently moving + bool newDir = (relative < 0); // CCW if <0, else set CW + if (ctrl.dir == newDir){ //current direction is the same + ctrl.statusPrev = ctrl.status; //update previous status + ctrl.status = ctrl.status==COAST ? COAST : ACC; //stay at coast otherwise switch to ACC + calc(abs(relative + ctrl.stepsRemaining)); //calculate new velolcity profile for new+remaining steps + } else { //direction has changed + //direction change not supported TODO wait for decel finish / queue? + STEP_LOGW("DendoStepper", "DIRECTION HOT-CHANGE NOT SUPPORTED - Finising previous move, this command will be ignored"); + return ESP_ERR_NOT_SUPPORTED; + } + } + else { //current state is IDLE + ctrl.statusPrev = ctrl.status; //update previous status + ctrl.status = ACC; + setDir(relative < 0); // set CCW if <0, else set CW + calc(abs(relative)); // calculate velocity profile + } + + currentPos += relative; //(target position / not actual) ESP_ERROR_CHECK(timer_set_alarm_value(conf.timer_group, conf.timer_idx, ctrl.stepInterval)); // set HW timer alarm to stepinterval + //TODO timer has to be stopped before update if already running? ESP_ERROR_CHECK(timer_start(conf.timer_group, conf.timer_idx)); // start the timer return ESP_OK; @@ -195,6 +209,37 @@ void DendoStepper::setSpeedMm(uint32_t speed, uint16_t accT, uint16_t decT) STEP_LOGI("DendoStepper", "Speed set: v=%d mm/s t+=%d s t-=%d s", speed, accT, decT); } +//CUSTOM - change speed while running +//FIXME: this approach does not work, since calc function would have to be run after change, this will mess up target steps... +//void DendoStepper::changeSpeed(uint32_t speed) +//{ +// //TODO reduce duplicate code (e.g. call setSpeed function) +// //change speed +// ctrl.speed = speed; +// //change status to ACC/DEC +// STEP_LOGI("DendoStepper", "Speed changed: from v=%.2f rad/s to v=%.2f rad/s", ctrl.speed, speed); +// if (speed > ctrl.speed) ctrl.status = ACC; +// if (speed < ctrl.speed) ctrl.status = DEC; +//} +// +//void DendoStepper::changeSpeedMm(uint32_t speed) +//{ +// //TODO reduce duplicate code (e.g. call setSpeedMm function) +// if (ctrl.stepsPerMm == 0) +// { +// STEP_LOGE("DendoStepper", "Steps per millimeter not set, cannot set the speed!"); +// } +// //calc new speed +// float speedNew = speed * ctrl.stepsPerMm; +// //change status to ACC/DEC +// if (speedNew > ctl.speed) ctrl.status = ACC; +// if (speedNew < ctl.speed) ctrl.status = DEC; +// //update speed, log output +// ctrl.speed = speedNew; +// STEP_LOGI("DendoStepper", "Speed changed: from v=%.2f rad/s to v=%.2f rad/s", ctrl.speed, speedNew); +//} + + void DendoStepper::setStepsPerMm(uint16_t steps) { ctrl.stepsPerMm = steps; @@ -261,6 +306,7 @@ void DendoStepper::stop() } ctrl.runInfinite = false; timer_pause(conf.timer_group, conf.timer_idx); // stop the timer + ctrl.statusPrev = ctrl.status; //update previous status ctrl.status = IDLE; ctrl.stepCnt = 0; gpio_set_level((gpio_num_t)conf.stepPin, 0); @@ -287,10 +333,20 @@ bool DendoStepper::xISR() ctrl.stepCnt++; + //CUSTOM: track actual precice current position + if (ctrl.dir) { + ctrl.posActual ++; + } else { + ctrl.posActual --; + } + //CUSTOM: track remaining steps for eventually resuming + ctrl.stepsRemaining = ctrl.stepCnt - ctrl.stepCnt; + // we are done if (ctrl.stepsToGo == ctrl.stepCnt && !ctrl.runInfinite) { timer_pause(conf.timer_group, conf.timer_idx); // stop the timer + ctrl.statusPrev = ctrl.status; //update previous status ctrl.status = IDLE; ctrl.stepCnt = 0; return 0; @@ -299,16 +355,19 @@ bool DendoStepper::xISR() if (ctrl.stepCnt > 0 && ctrl.stepCnt < ctrl.accEnd) { // we are accelerating ctrl.currentSpeed += ctrl.accInc; + ctrl.statusPrev = ctrl.status; //update previous status ctrl.status = ACC; // we are accelerating, note that*/ } else if (ctrl.stepCnt > ctrl.coastEnd && !ctrl.runInfinite) { // we must be deccelerating then ctrl.currentSpeed -= ctrl.decInc; + ctrl.statusPrev = ctrl.status; //update previous status ctrl.status = DEC; // we are deccelerating } else { ctrl.currentSpeed = ctrl.targetSpeed; + ctrl.statusPrev = ctrl.status; //update previous status ctrl.status = COAST; // we are coasting } @@ -316,15 +375,18 @@ bool DendoStepper::xISR() // set alarm to calculated interval and disable pin GPIO.out_w1tc = (1ULL << conf.stepPin); timer_set_alarm_value(conf.timer_group, conf.timer_idx, ctrl.stepInterval); + ctrl.stepCnt++; return 1; } void DendoStepper::calc(uint32_t targetSteps) { - - ctrl.accSteps = 0.5 * ctrl.acc * (ctrl.speed / ctrl.acc) * (ctrl.speed / ctrl.acc); - + //steps from ctrl.speed -> 0: ctrl.decSteps = 0.5 * ctrl.dec * (ctrl.speed / ctrl.dec) * (ctrl.speed / ctrl.dec); + //steps from 0 -> ctrl.speed: + //ctrl.accSteps = 0.5 * ctrl.acc * (ctrl.speed / ctrl.acc) * (ctrl.speed / ctrl.acc); + //steps from ctrl.currentSpeed -> ctrl.speed: + ctrl.accSteps = 0.5 * ctrl.acc * (ctrl.speed / ctrl.acc) * (ctrl.speed / ctrl.acc) * (ctrl.speed - ctrl.currentSpeed) / ctrl.speed; if (targetSteps < (ctrl.decSteps + ctrl.accSteps)) { diff --git a/components/DendoStepper/include/DendoStepper.h b/components/DendoStepper/include/DendoStepper.h index 9846412..a327ecc 100644 --- a/components/DendoStepper/include/DendoStepper.h +++ b/components/DendoStepper/include/DendoStepper.h @@ -94,6 +94,9 @@ typedef struct float dec = 100; // decceleration in rad*second^-2 uint32_t accSteps = 0; uint32_t decSteps = 0; + int32_t stepsRemaining = 0; + uint64_t posActual = 0; //actual current pos incremented at every step + uint8_t statusPrev = DISABLED; //FIXME currently unused uint8_t status = DISABLED; bool dir = CW; bool runInfinite = false; @@ -208,6 +211,9 @@ public: */ void setSpeedMm(uint32_t speed, uint16_t accT, uint16_t decT); + //CUSTOM: change speed while running + void changeSpeedMm(uint32_t speed); + /** * @brief Set steps per 1 mm of linear movement * @@ -272,4 +278,4 @@ public: void stop(); }; -#endif \ No newline at end of file +#endif diff --git a/main/guide-stepper.cpp b/main/guide-stepper.cpp index 37cd5de..8dbd96b 100644 --- a/main/guide-stepper.cpp +++ b/main/guide-stepper.cpp @@ -50,6 +50,7 @@ static DendoStepper step; static const char *TAG = "stepper"; //tag for logging static bool stepp_direction = true; +static bool dir = true, dirPrev; //TODO local variables in travelSteps? static uint32_t posNow = 0; @@ -65,13 +66,23 @@ void travelSteps(int stepsTarget){ stepsToGo = abs(stepsTarget); if(stepsTarget < 0) stepp_direction = !stepp_direction; //invert direction in reverse mode + while (stepsToGo != 0){ + + //--- wait if direction changed --- + if (dirPrev != dir){ + ESP_LOGI(TAG, " dir-change detected - waiting for move to finish \n "); + while(step.getState() != 1) vTaskDelay(1); //wait for move to finish + } + //--- currently moving right --- if (stepp_direction == true){ //currently moving right remaining = POS_MAX_STEPS - posNow; //calc remaining distance fom current position to limit if (stepsToGo > remaining){ //new distance will exceed limit step.runAbs (POS_MAX_STEPS); //move to limit - while(step.getState() != 1) vTaskDelay(1); //wait for move to finish + dirPrev = dir; + dir = 1; + //while(step.getState() != 1) vTaskDelay(1); //wait for move to finish posNow = POS_MAX_STEPS; stepp_direction = false; //change current direction for next iteration stepsToGo = stepsToGo - remaining; //decrease target length by already traveled distance @@ -79,7 +90,10 @@ void travelSteps(int stepsTarget){ } else { //target distance does not reach the limit step.runAbs (posNow + stepsToGo); //move by (remaining) distance to reach target length - while(step.getState() != 1) vTaskDelay(1); //wait for move to finish + dirPrev = dir; + dir = 1; + //-- dont wait for move to finish since moves in same direction get merged -- + //while(step.getState() != 1) vTaskDelay(1); //wait for move to finish ESP_LOGD(TAG, "moving to %d\n", posNow+stepsToGo); posNow += stepsToGo; stepsToGo = 0; //finished, reset target length (could as well exit loop/break) @@ -91,7 +105,9 @@ void travelSteps(int stepsTarget){ remaining = posNow - POS_MIN_STEPS; if (stepsToGo > remaining){ step.runAbs (POS_MIN_STEPS); - while(step.getState() != 1) vTaskDelay(2); //wait for move to finish + dirPrev = dir; + dir = 0; + //while(step.getState() != 1) vTaskDelay(2); //wait for move to finish posNow = POS_MIN_STEPS; stepp_direction = true; stepsToGo = stepsToGo - remaining; @@ -99,7 +115,10 @@ void travelSteps(int stepsTarget){ } else { step.runAbs (posNow - stepsToGo); //when moving left the coordinate has to be decreased - while(step.getState() != 1) vTaskDelay(2); //wait for move to finish + dirPrev = dir; + dir = 0; + //-- dont wait for move to finish since moves in same direction get merged -- + //while(step.getState() != 1) vTaskDelay(2); //wait for move to finish ESP_LOGD(TAG, "moving to %d\n", posNow - stepsToGo); posNow -= stepsToGo; stepsToGo = 0; From 45409676a0b502c69b5bc26be7361bfa5a89ed61 Mon Sep 17 00:00:00 2001 From: jonny_ji7 Date: Mon, 13 Mar 2023 11:12:42 +0100 Subject: [PATCH 2/8] Stepper: add mutex, logging, delay, less steps When testing last commit the stepper task crashed almost instantly and moved randomly. Trying to fix/debug that with those changes --- components/DendoStepper/DendoStepper.cpp | 20 ++++++++++++------- .../DendoStepper/include/DendoStepper.h | 4 +++- main/guide-stepper.cpp | 6 ++++-- 3 files changed, 20 insertions(+), 10 deletions(-) diff --git a/components/DendoStepper/DendoStepper.cpp b/components/DendoStepper/DendoStepper.cpp index 07d31aa..d053874 100644 --- a/components/DendoStepper/DendoStepper.cpp +++ b/components/DendoStepper/DendoStepper.cpp @@ -128,10 +128,13 @@ esp_err_t DendoStepper::runPos(int32_t relative) if (ctrl.dir == newDir){ //current direction is the same ctrl.statusPrev = ctrl.status; //update previous status ctrl.status = ctrl.status==COAST ? COAST : ACC; //stay at coast otherwise switch to ACC + xSemaphoreTake(semaphore_isrVariables, portMAX_DELAY); calc(abs(relative + ctrl.stepsRemaining)); //calculate new velolcity profile for new+remaining steps + ESP_LOGW("DendoStepper", "EXTEND running movement (stepsRemaining: %d + stepsNew: %d)", ctrl.stepsRemaining, relative); + xSemaphoreGive(semaphore_isrVariables); } else { //direction has changed //direction change not supported TODO wait for decel finish / queue? - STEP_LOGW("DendoStepper", "DIRECTION HOT-CHANGE NOT SUPPORTED - Finising previous move, this command will be ignored"); + STEP_LOGE("DendoStepper", "DIRECTION HOT-CHANGE NOT SUPPORTED - Finising previous move, this command will be ignored"); return ESP_ERR_NOT_SUPPORTED; } } @@ -333,14 +336,17 @@ bool DendoStepper::xISR() ctrl.stepCnt++; - //CUSTOM: track actual precice current position - if (ctrl.dir) { - ctrl.posActual ++; - } else { - ctrl.posActual --; - } + ////CUSTOM: track actual precice current position + //if (ctrl.dir) { + // ctrl.posActual ++; + //} else { + // ctrl.posActual --; + //} + //CUSTOM: track remaining steps for eventually resuming + xSemaphoreTake(semaphore_isrVariables, portMAX_DELAY); ctrl.stepsRemaining = ctrl.stepCnt - ctrl.stepCnt; + xSemaphoreGive(semaphore_isrVariables); // we are done if (ctrl.stepsToGo == ctrl.stepCnt && !ctrl.runInfinite) diff --git a/components/DendoStepper/include/DendoStepper.h b/components/DendoStepper/include/DendoStepper.h index a327ecc..9b535ad 100644 --- a/components/DendoStepper/include/DendoStepper.h +++ b/components/DendoStepper/include/DendoStepper.h @@ -29,6 +29,7 @@ #include "freertos/task.h" #include "esp_timer.h" #include "math.h" +#include "freertos/semphr.h" //#define STEP_DEBUG @@ -95,7 +96,7 @@ typedef struct uint32_t accSteps = 0; uint32_t decSteps = 0; int32_t stepsRemaining = 0; - uint64_t posActual = 0; //actual current pos incremented at every step + //uint64_t posActual = 0; //actual current pos incremented at every step uint8_t statusPrev = DISABLED; //FIXME currently unused uint8_t status = DISABLED; bool dir = CW; @@ -111,6 +112,7 @@ private: ctrl_var_t ctrl; esp_timer_handle_t dyingTimer; TaskHandle_t enTask; + SemaphoreHandle_t semaphore_isrVariables = NULL; uint64_t currentPos = 0; // absolute position bool timerStarted = 0; diff --git a/main/guide-stepper.cpp b/main/guide-stepper.cpp index 8dbd96b..e70c890 100644 --- a/main/guide-stepper.cpp +++ b/main/guide-stepper.cpp @@ -34,7 +34,7 @@ extern "C" #define ACCEL_MS 100.0 //ms from 0 to max #define DECEL_MS 90.0 //ms from max to 0 -#define STEPPER_STEPS_PER_ROT 1600 +#define STEPPER_STEPS_PER_ROT 800 #define STEPPER_STEPS_PER_MM STEPPER_STEPS_PER_ROT/4 #define D_CABLE 6 @@ -71,7 +71,7 @@ void travelSteps(int stepsTarget){ //--- wait if direction changed --- if (dirPrev != dir){ - ESP_LOGI(TAG, " dir-change detected - waiting for move to finish \n "); + ESP_LOGW(TAG, " dir-change detected - waiting for move to finish \n "); while(step.getState() != 1) vTaskDelay(1); //wait for move to finish } @@ -235,6 +235,8 @@ void task_stepper_ctl(void *pvParameter) //read potentiometer and normalize (0-1) to get a variable for testing potiModifier = (float) gpio_readAdc(ADC_CHANNEL_POTI) / 4095; //0-4095 -> 0-1 //ESP_LOGI(TAG, "current poti-modifier = %f", potiModifier); + ESP_LOGI(TAG, "delaying stepper-ctl task by %.1f ms (poti value)", 2000 * potiModifier); + vTaskDelay(2000 * potiModifier / portTICK_PERIOD_MS); //calculate steps to move cableLen = (double)encStepsDelta * 1000 / ENCODER_STEPS_PER_METER; From 0a05340763c26be4735d1cf845b96150112b25ad Mon Sep 17 00:00:00 2001 From: jonny_l480 Date: Mon, 13 Mar 2023 18:26:50 +0100 Subject: [PATCH 3/8] revert mutex, fix state reset, logging, slower stepsRemaining not necessary in ISR comment out forced stop at runAbs statusPrev not needed TODO: ISR defines state every time, no need to adjust manually while running adjust calc function to handle remaining steps Test: broken - slow moves mork but when "extending" movement it just vibrates --- components/DendoStepper/DendoStepper.cpp | 55 +++++++++++++----------- main/guide-stepper.cpp | 11 ++--- main/main.cpp | 2 +- 3 files changed, 37 insertions(+), 31 deletions(-) diff --git a/components/DendoStepper/DendoStepper.cpp b/components/DendoStepper/DendoStepper.cpp index d053874..53878c9 100644 --- a/components/DendoStepper/DendoStepper.cpp +++ b/components/DendoStepper/DendoStepper.cpp @@ -55,6 +55,9 @@ void DendoStepper::init(uint8_t stepP, uint8_t dirP, uint8_t enP, timer_group_t void DendoStepper::init() { + ESP_LOGW("DendoStepper", "semaphore init"); + semaphore_isrVariables = xSemaphoreCreateBinary(); + xSemaphoreGive(semaphore_isrVariables); uint64_t mask = (1ULL << conf.stepPin) | (1ULL << conf.dirPin) | (1ULL << conf.enPin); // put output gpio pins in bitmask gpio_config_t gpio_conf = { // config gpios @@ -120,18 +123,19 @@ esp_err_t DendoStepper::runPos(int32_t relative) if (ctrl.status == DISABLED) // if motor is disabled, enable it enableMotor(); + bool newDir = (relative < 0); // CCW if <0, else set CW + printf("runpos start -- steps: %d, remaining: %d, status: %d, olddir: %d, newdir: %d\n", relative, ctrl.stepsRemaining, ctrl.status, ctrl.dir, newDir); if (!relative) // why would u call it with 0 wtf return ESP_ERR_NOT_SUPPORTED; if (ctrl.status > IDLE) { //currently moving - bool newDir = (relative < 0); // CCW if <0, else set CW if (ctrl.dir == newDir){ //current direction is the same - ctrl.statusPrev = ctrl.status; //update previous status + //ctrl.statusPrev = ctrl.status; //update previous status ctrl.status = ctrl.status==COAST ? COAST : ACC; //stay at coast otherwise switch to ACC - xSemaphoreTake(semaphore_isrVariables, portMAX_DELAY); + ctrl.stepsRemaining = ctrl.stepsToGo - ctrl.stepCnt; calc(abs(relative + ctrl.stepsRemaining)); //calculate new velolcity profile for new+remaining steps ESP_LOGW("DendoStepper", "EXTEND running movement (stepsRemaining: %d + stepsNew: %d)", ctrl.stepsRemaining, relative); - xSemaphoreGive(semaphore_isrVariables); + ESP_ERROR_CHECK(timer_set_alarm_value(conf.timer_group, conf.timer_idx, ctrl.stepInterval)); // set HW timer alarm to stepinterval } else { //direction has changed //direction change not supported TODO wait for decel finish / queue? STEP_LOGE("DendoStepper", "DIRECTION HOT-CHANGE NOT SUPPORTED - Finising previous move, this command will be ignored"); @@ -139,17 +143,16 @@ esp_err_t DendoStepper::runPos(int32_t relative) } } else { //current state is IDLE - ctrl.statusPrev = ctrl.status; //update previous status + //ctrl.statusPrev = ctrl.status; //update previous status ctrl.status = ACC; setDir(relative < 0); // set CCW if <0, else set CW calc(abs(relative)); // calculate velocity profile + ESP_ERROR_CHECK(timer_set_alarm_value(conf.timer_group, conf.timer_idx, ctrl.stepInterval)); // set HW timer alarm to stepinterval + ESP_ERROR_CHECK(timer_start(conf.timer_group, conf.timer_idx)); // start the timer } + printf("runpos end -- steps: %d, status: %d, olddir: %d, newdir: %d\n", relative, ctrl.status, ctrl.dir, newDir); currentPos += relative; //(target position / not actual) - ESP_ERROR_CHECK(timer_set_alarm_value(conf.timer_group, conf.timer_idx, ctrl.stepInterval)); // set HW timer alarm to stepinterval - //TODO timer has to be stopped before update if already running? - ESP_ERROR_CHECK(timer_start(conf.timer_group, conf.timer_idx)); // start the timer - return ESP_OK; } @@ -164,13 +167,13 @@ esp_err_t DendoStepper::runPosMm(int32_t relative) esp_err_t DendoStepper::runAbs(uint32_t position) { - if (getState() > IDLE) // we are already moving, so stop it - stop(); - while (getState() > IDLE) - { - // waiting for idle, watchdog should take care of inf loop if it occurs - vTaskDelay(1); - } // shouldnt take long tho + //if (getState() > IDLE) // we are already moving, so stop it + // stop(); + //while (getState() > IDLE) + //{ + // // waiting for idle, watchdog should take care of inf loop if it occurs + // vTaskDelay(1); + //} // shouldnt take long tho if (position == currentPos) // we cant go anywhere return 0; @@ -309,7 +312,7 @@ void DendoStepper::stop() } ctrl.runInfinite = false; timer_pause(conf.timer_group, conf.timer_idx); // stop the timer - ctrl.statusPrev = ctrl.status; //update previous status + //ctrl.statusPrev = ctrl.status; //update previous status ctrl.status = IDLE; ctrl.stepCnt = 0; gpio_set_level((gpio_num_t)conf.stepPin, 0); @@ -344,15 +347,15 @@ bool DendoStepper::xISR() //} //CUSTOM: track remaining steps for eventually resuming - xSemaphoreTake(semaphore_isrVariables, portMAX_DELAY); - ctrl.stepsRemaining = ctrl.stepCnt - ctrl.stepCnt; - xSemaphoreGive(semaphore_isrVariables); + //xSemaphoreTake(semaphore_isrVariables, portMAX_DELAY); + //ctrl.stepsRemaining = ctrl.stepCnt - ctrl.stepCnt; + //xSemaphoreGive(semaphore_isrVariables); // we are done if (ctrl.stepsToGo == ctrl.stepCnt && !ctrl.runInfinite) { timer_pause(conf.timer_group, conf.timer_idx); // stop the timer - ctrl.statusPrev = ctrl.status; //update previous status + //ctrl.statusPrev = ctrl.status; //update previous status ctrl.status = IDLE; ctrl.stepCnt = 0; return 0; @@ -361,19 +364,19 @@ bool DendoStepper::xISR() if (ctrl.stepCnt > 0 && ctrl.stepCnt < ctrl.accEnd) { // we are accelerating ctrl.currentSpeed += ctrl.accInc; - ctrl.statusPrev = ctrl.status; //update previous status + //ctrl.statusPrev = ctrl.status; //update previous status ctrl.status = ACC; // we are accelerating, note that*/ } else if (ctrl.stepCnt > ctrl.coastEnd && !ctrl.runInfinite) { // we must be deccelerating then ctrl.currentSpeed -= ctrl.decInc; - ctrl.statusPrev = ctrl.status; //update previous status + //ctrl.statusPrev = ctrl.status; //update previous status ctrl.status = DEC; // we are deccelerating } else { ctrl.currentSpeed = ctrl.targetSpeed; - ctrl.statusPrev = ctrl.status; //update previous status + //ctrl.statusPrev = ctrl.status; //update previous status ctrl.status = COAST; // we are coasting } @@ -381,12 +384,14 @@ bool DendoStepper::xISR() // set alarm to calculated interval and disable pin GPIO.out_w1tc = (1ULL << conf.stepPin); timer_set_alarm_value(conf.timer_group, conf.timer_idx, ctrl.stepInterval); - ctrl.stepCnt++; return 1; } void DendoStepper::calc(uint32_t targetSteps) { + //CUSTOM reset counter if already moving + ctrl.stepCnt = 0; + //steps from ctrl.speed -> 0: ctrl.decSteps = 0.5 * ctrl.dec * (ctrl.speed / ctrl.dec) * (ctrl.speed / ctrl.dec); //steps from 0 -> ctrl.speed: diff --git a/main/guide-stepper.cpp b/main/guide-stepper.cpp index e70c890..d225710 100644 --- a/main/guide-stepper.cpp +++ b/main/guide-stepper.cpp @@ -23,7 +23,7 @@ extern "C" #define STEPPER_TEST_TRAVEL 65 //mm // #define MIN_MM 0 -#define MAX_MM 60 +#define MAX_MM 110 //60 #define POS_MAX_STEPS MAX_MM * STEPPER_STEPS_PER_MM #define POS_MIN_STEPS MIN_MM * STEPPER_STEPS_PER_MM @@ -31,10 +31,11 @@ extern "C" #define SPEED_MIN 2.0 //mm/s #define SPEED_MAX 60.0 //mm/s -#define ACCEL_MS 100.0 //ms from 0 to max -#define DECEL_MS 90.0 //ms from max to 0 +#define SPEED 10 //35, 100, 50 rev +#define ACCEL_MS 600.0 //ms from 0 to max +#define DECEL_MS 1000.0 //ms from max to 0 -#define STEPPER_STEPS_PER_ROT 800 +#define STEPPER_STEPS_PER_ROT 1600 #define STEPPER_STEPS_PER_MM STEPPER_STEPS_PER_ROT/4 #define D_CABLE 6 @@ -252,7 +253,7 @@ void task_stepper_ctl(void *pvParameter) ESP_LOGD(TAG, "cablelen=%.2lf, turns=%.2lf, travelMm=%.3lf, travelStepsExact: %.3lf, travelStepsFull=%d, partialStep=%.3lf", cableLen, turns, travelMm, travelStepsExact, travelStepsFull, travelStepsPartial); ESP_LOGI(TAG, "MOVING %d steps", travelStepsFull); //TODO: calculate variable speed for smoother movement? for example intentionally lag behind and calculate speed according to buffered data - step.setSpeedMm(35, 100, 50); + step.setSpeedMm(SPEED, ACCEL_MS, DECEL_MS); //testing: get speed from poti //step.setSpeedMm(35, 1000*potiModifier+1, 1000*potiModifier+1); travelSteps(travelStepsExact); diff --git a/main/main.cpp b/main/main.cpp index 7872214..45cd687 100644 --- a/main/main.cpp +++ b/main/main.cpp @@ -98,7 +98,7 @@ extern "C" void app_main() xTaskCreate(task_control, "task_control", configMINIMAL_STACK_SIZE * 3, NULL, 5, NULL); //create task for controlling the machine - xTaskCreate(task_stepper_ctl, "task_stepper_ctl", configMINIMAL_STACK_SIZE * 3, NULL, 5, NULL); + xTaskCreate(task_stepper_ctl, "task_stepper_ctl", configMINIMAL_STACK_SIZE * 5, NULL, 5, NULL); //create task for handling the buzzer xTaskCreate(&task_buzzer, "task_buzzer", 2048, NULL, 2, NULL); From dc6deeb3d09628ac7f4f4fab256dc168bcbf835a Mon Sep 17 00:00:00 2001 From: jonny_ji7 Date: Mon, 13 Mar 2023 20:02:22 +0100 Subject: [PATCH 4/8] Stepper: rework resume functionality --- components/DendoStepper/DendoStepper.cpp | 70 ++++++++++++------------ main/guide-stepper.cpp | 8 +-- 2 files changed, 39 insertions(+), 39 deletions(-) diff --git a/components/DendoStepper/DendoStepper.cpp b/components/DendoStepper/DendoStepper.cpp index 53878c9..f77b203 100644 --- a/components/DendoStepper/DendoStepper.cpp +++ b/components/DendoStepper/DendoStepper.cpp @@ -117,41 +117,35 @@ timer_avail: ESP_ERROR_CHECK(timer_isr_callback_add(conf.timer_group, conf.timer_idx, xISRwrap, this, 0)); // add callback fn to run when alarm is triggrd } + esp_err_t DendoStepper::runPos(int32_t relative) { //TODO only enable when actually moving if (ctrl.status == DISABLED) // if motor is disabled, enable it enableMotor(); - bool newDir = (relative < 0); // CCW if <0, else set CW - printf("runpos start -- steps: %d, remaining: %d, status: %d, olddir: %d, newdir: %d\n", relative, ctrl.stepsRemaining, ctrl.status, ctrl.dir, newDir); + setDir(relative < 0); // set CCW if <0, else set CW + if (!relative) // why would u call it with 0 wtf return ESP_ERR_NOT_SUPPORTED; if (ctrl.status > IDLE) { //currently moving - if (ctrl.dir == newDir){ //current direction is the same - //ctrl.statusPrev = ctrl.status; //update previous status - ctrl.status = ctrl.status==COAST ? COAST : ACC; //stay at coast otherwise switch to ACC - ctrl.stepsRemaining = ctrl.stepsToGo - ctrl.stepCnt; - calc(abs(relative + ctrl.stepsRemaining)); //calculate new velolcity profile for new+remaining steps - ESP_LOGW("DendoStepper", "EXTEND running movement (stepsRemaining: %d + stepsNew: %d)", ctrl.stepsRemaining, relative); - ESP_ERROR_CHECK(timer_set_alarm_value(conf.timer_group, conf.timer_idx, ctrl.stepInterval)); // set HW timer alarm to stepinterval - } else { //direction has changed - //direction change not supported TODO wait for decel finish / queue? - STEP_LOGE("DendoStepper", "DIRECTION HOT-CHANGE NOT SUPPORTED - Finising previous move, this command will be ignored"); - return ESP_ERR_NOT_SUPPORTED; - } - } - else { //current state is IDLE - //ctrl.statusPrev = ctrl.status; //update previous status - ctrl.status = ACC; - setDir(relative < 0); // set CCW if <0, else set CW - calc(abs(relative)); // calculate velocity profile - ESP_ERROR_CHECK(timer_set_alarm_value(conf.timer_group, conf.timer_idx, ctrl.stepInterval)); // set HW timer alarm to stepinterval - ESP_ERROR_CHECK(timer_start(conf.timer_group, conf.timer_idx)); // start the timer + ctrl.status = ctrl.status==COAST ? COAST : ACC; //stay at coast otherwise switch to ACC + ctrl.stepsRemaining = ctrl.stepsToGo - ctrl.stepCnt; + calc(abs(relative + ctrl.stepsRemaining)); //calculate new velolcity profile for new+remaining steps + ESP_LOGW("DendoStepper", "EXTEND running movement (stepsRemaining: %d + stepsNew: %d)", ctrl.stepsRemaining, relative); + ESP_ERROR_CHECK(timer_set_alarm_value(conf.timer_group, conf.timer_idx, ctrl.stepInterval)); // set HW timer alarm to stepinterval } - printf("runpos end -- steps: %d, status: %d, olddir: %d, newdir: %d\n", relative, ctrl.status, ctrl.dir, newDir); + else { //current state is IDLE + //ctrl.statusPrev = ctrl.status; //update previous status + ctrl.status = ACC; + calc(abs(relative)); // calculate velocity profile + ESP_ERROR_CHECK(timer_set_alarm_value(conf.timer_group, conf.timer_idx, ctrl.stepInterval)); // set HW timer alarm to stepinterval + ESP_ERROR_CHECK(timer_start(conf.timer_group, conf.timer_idx)); // start the timer + } + + //printf("runpos end -- steps: %d, status: %d, olddir: %d, newdir: %d\n", relative, ctrl.status, ctrl.dir, newDir); currentPos += relative; //(target position / not actual) return ESP_OK; } @@ -165,26 +159,31 @@ esp_err_t DendoStepper::runPosMm(int32_t relative) return runPos(relative * ctrl.stepsPerMm); } +//customized: if already running and direction is the same immediately pass to runPos esp_err_t DendoStepper::runAbs(uint32_t position) { - //if (getState() > IDLE) // we are already moving, so stop it - // stop(); - //while (getState() > IDLE) - //{ - // // waiting for idle, watchdog should take care of inf loop if it occurs - // vTaskDelay(1); - //} // shouldnt take long tho - - if (position == currentPos) // we cant go anywhere - return 0; + //exit if nothing to do + if (position == currentPos) return 0; //already at position + //calculate steps necessary int32_t relativeSteps = 0; relativeSteps = (int32_t)position - currentPos; + //wait if direction needs to change + if (getState() > IDLE){//already moving + bool newDir = (relativeSteps < 0); // CCW if <0, else set CW + if (ctrl.dir != newDir){ //direction differs + STEP_LOGE("DendoStepper", "DIRECTION HOT-CHANGE NOT SUPPORTED - Waiting for move to finish..."); + while (getState() > IDLE) vTaskDelay(1); //wait for move to finish + } + } + + //call runPos with new target position ESP_LOGI("DendoStepper", "Cur: %llu move %d", currentPos, relativeSteps); return runPos(relativeSteps); // run to new position } + esp_err_t DendoStepper::runAbsMm(uint32_t position) { if (ctrl.stepsPerMm == 0) @@ -404,7 +403,8 @@ void DendoStepper::calc(uint32_t targetSteps) ESP_LOGI("Dendostepper", "Computing new speed"); ctrl.speed = sqrt(2 * targetSteps * ((ctrl.dec * ctrl.acc) / (ctrl.dec + ctrl.acc))); - ctrl.accSteps = 0.5 * ctrl.acc * (ctrl.speed / ctrl.acc) * (ctrl.speed / ctrl.acc); + //ctrl.accSteps = 0.5 * ctrl.acc * (ctrl.speed / ctrl.acc) * (ctrl.speed / ctrl.acc); + ctrl.accSteps = 0.5 * ctrl.acc * (ctrl.speed / ctrl.acc) * (ctrl.speed / ctrl.acc) * (ctrl.speed - ctrl.currentSpeed) / ctrl.speed; ctrl.decSteps = 0.5 * ctrl.dec * (ctrl.speed / ctrl.dec) * (ctrl.speed / ctrl.dec); } @@ -412,7 +412,7 @@ void DendoStepper::calc(uint32_t targetSteps) ctrl.coastEnd = targetSteps - ctrl.decSteps; ctrl.targetSpeed = ctrl.speed; - ctrl.accInc = ctrl.targetSpeed / (double)ctrl.accSteps; + ctrl.accInc = (ctrl.targetSpeed - ctrl.currentSpeed) / (double)ctrl.accSteps; ctrl.decInc = ctrl.targetSpeed / (double)ctrl.decSteps; ctrl.currentSpeed = ctrl.accInc; diff --git a/main/guide-stepper.cpp b/main/guide-stepper.cpp index d225710..3542176 100644 --- a/main/guide-stepper.cpp +++ b/main/guide-stepper.cpp @@ -71,10 +71,10 @@ void travelSteps(int stepsTarget){ while (stepsToGo != 0){ //--- wait if direction changed --- - if (dirPrev != dir){ - ESP_LOGW(TAG, " dir-change detected - waiting for move to finish \n "); - while(step.getState() != 1) vTaskDelay(1); //wait for move to finish - } + //if (dirPrev != dir){ + // ESP_LOGW(TAG, " dir-change detected - waiting for move to finish \n "); + // while(step.getState() != 1) vTaskDelay(1); //wait for move to finish + //} //--- currently moving right --- if (stepp_direction == true){ //currently moving right From 8d2428d28576e08da07af79659620376b1e4e964 Mon Sep 17 00:00:00 2001 From: jonny_l480 Date: Mon, 13 Mar 2023 20:25:10 +0100 Subject: [PATCH 5/8] Stepper: Add calc debug output ISSUES - sometimes deadlock at direction change waits for idle state in runAbs function, but stepper is not really moving anymore ISR does not change to idle or state changed afterwards? - every EXTEND operation results in a motor stop thus when extending alot is stopped only ramping up and down when encoder stop after several extend attempts the stepper moves all the remaining steps successfully --- components/DendoStepper/DendoStepper.cpp | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/components/DendoStepper/DendoStepper.cpp b/components/DendoStepper/DendoStepper.cpp index f77b203..33126f5 100644 --- a/components/DendoStepper/DendoStepper.cpp +++ b/components/DendoStepper/DendoStepper.cpp @@ -130,7 +130,7 @@ esp_err_t DendoStepper::runPos(int32_t relative) return ESP_ERR_NOT_SUPPORTED; if (ctrl.status > IDLE) { //currently moving - ctrl.status = ctrl.status==COAST ? COAST : ACC; //stay at coast otherwise switch to ACC + //ctrl.status = ctrl.status==COAST ? COAST : ACC; //stay at coast otherwise switch to ACC ctrl.stepsRemaining = ctrl.stepsToGo - ctrl.stepCnt; calc(abs(relative + ctrl.stepsRemaining)); //calculate new velolcity profile for new+remaining steps ESP_LOGW("DendoStepper", "EXTEND running movement (stepsRemaining: %d + stepsNew: %d)", ctrl.stepsRemaining, relative); @@ -420,5 +420,7 @@ void DendoStepper::calc(uint32_t targetSteps) ctrl.stepInterval = TIMER_F / ctrl.currentSpeed; ctrl.stepsToGo = targetSteps; + printf("CALC: speedNow=%.1f, speedTarget=%.1f, accEnd=%d, coastEnd=%d, accSteps=%d, accInc=%.3f\n", + ctrl.currentSpeed, ctrl.targetSpeed, ctrl.accEnd, ctrl.coastEnd, ctrl.accSteps, ctrl.accInc); STEP_LOGI("calc", "acc end:%u coastend:%u stepstogo:%u speed:%f acc:%f int: %u", ctrl.accEnd, ctrl.coastEnd, ctrl.stepsToGo, ctrl.speed, ctrl.acc, ctrl.stepInterval); } From 5d7c67ab9a2327adc49ea92cd55f9aa23a40b065 Mon Sep 17 00:00:00 2001 From: jonny_l480 Date: Fri, 24 Mar 2023 12:10:37 +0100 Subject: [PATCH 6/8] stepper-test: Control stepper with buttons now the stepper test task controls the stepper using button input this makes debugging modifications to stepper library easier --- main/config.hpp | 2 +- main/guide-stepper.cpp | 60 ++++++++++++++++++++++++++++++++++-------- main/main.cpp | 2 +- 3 files changed, 51 insertions(+), 13 deletions(-) diff --git a/main/config.hpp b/main/config.hpp index dcd474b..95ca28b 100644 --- a/main/config.hpp +++ b/main/config.hpp @@ -85,7 +85,7 @@ extern "C" { //----- stepper config ----- //-------------------------- //enable stepper test mode (dont start control and encoder task) -//#define STEPPER_TEST +#define STEPPER_TEST #define STEPPER_STEP_PIN GPIO_NUM_18 //mos1 #define STEPPER_DIR_PIN GPIO_NUM_16 //ST3 #define STEPPER_EN_PIN GPIO_NUM_0 //not connected (-> stepper always on) diff --git a/main/guide-stepper.cpp b/main/guide-stepper.cpp index 3542176..fb6605b 100644 --- a/main/guide-stepper.cpp +++ b/main/guide-stepper.cpp @@ -11,6 +11,7 @@ extern "C" #include "config.hpp" #include "guide-stepper.hpp" #include "encoder.hpp" +#include "gpio_evaluateSwitch.hpp" @@ -32,7 +33,7 @@ extern "C" #define SPEED_MAX 60.0 //mm/s #define SPEED 10 //35, 100, 50 rev -#define ACCEL_MS 600.0 //ms from 0 to max +#define ACCEL_MS 1000.0 //ms from 0 to max #define DECEL_MS 1000.0 //ms from max to 0 #define STEPPER_STEPS_PER_ROT 1600 @@ -188,18 +189,55 @@ void task_stepper_test(void *pvParameter) { init_stepper(); home(); + step.setSpeedMm(SPEED, ACCEL_MS, DECEL_MS); + //--- move from left to right repeatedly --- + // while (1) { + // updateSpeedFromAdc(); + // step.runPosMm(STEPPER_TEST_TRAVEL); + // while(step.getState() != 1) vTaskDelay(2); + // ESP_LOGI(TAG, "finished moving right => moving left"); - while (1) { - updateSpeedFromAdc(); - step.runPosMm(STEPPER_TEST_TRAVEL); - while(step.getState() != 1) vTaskDelay(2); - ESP_LOGI(TAG, "finished moving right => moving left"); + // 10updateSpeedFromAdc(); + // step.runPosMm(-STEPPER_TEST_TRAVEL); + // while(step.getState() != 1) vTaskDelay(2); //1=idle + // ESP_LOGI(TAG, "finished moving left => moving right"); + // } - updateSpeedFromAdc(); - step.runPosMm(-STEPPER_TEST_TRAVEL); - while(step.getState() != 1) vTaskDelay(2); //1=idle - ESP_LOGI(TAG, "finished moving left => moving right"); - } + //--- control stepper using preset buttons --- + 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(); + + if (SW_RESET.risingEdge) { + buzzer.beep(1, 1000, 100); + step.stop(); + ESP_LOGI(TAG, "button - stop stepper\n "); + } + if (SW_PRESET1.risingEdge) { + buzzer.beep(2, 300, 100); + step.runPosMm(30); + ESP_LOGI(TAG, "button - moving right\n "); + } + if (SW_PRESET3.risingEdge) { + buzzer.beep(1, 500, 100); + step.runPosMm(-30); + ESP_LOGI(TAG, "button - moving left\n "); + } + if (SW_PRESET2.risingEdge) { + buzzer.beep(1, 100, 100); + ESP_LOGW(TAG, "button - current state: %d\n, pos: %llu", (int)step.getState(), step.getPositionMm()); + } + } } diff --git a/main/main.cpp b/main/main.cpp index 45cd687..3b7494b 100644 --- a/main/main.cpp +++ b/main/main.cpp @@ -99,10 +99,10 @@ extern "C" void app_main() //create task for controlling the machine xTaskCreate(task_stepper_ctl, "task_stepper_ctl", configMINIMAL_STACK_SIZE * 5, NULL, 5, NULL); +#endif //create task for handling the buzzer xTaskCreate(&task_buzzer, "task_buzzer", 2048, NULL, 2, NULL); -#endif //beep at startup buzzer.beep(3, 70, 50); From 1c59e0097b3732d9b035881faa442349c50d4341 Mon Sep 17 00:00:00 2001 From: jonny_l480 Date: Fri, 24 Mar 2023 15:00:39 +0100 Subject: [PATCH 7/8] Fix feature: extend movement works now! (test mode) while currently in stepper test mode the extend feature works as intended now. you can trigger movement using the buttons and repeating presses while still moving (coast, accel or decel) extends the movement without stopping or accelerating again. Note reversing the direction while still moving is not handled and results in immediate direction change and distance that makes no sense. also added alot of debug output and switched to debug level --- components/DendoStepper/DendoStepper.cpp | 30 +++++++++++++++++++----- main/guide-stepper.cpp | 16 +++++++------ main/main.cpp | 2 +- 3 files changed, 34 insertions(+), 14 deletions(-) diff --git a/components/DendoStepper/DendoStepper.cpp b/components/DendoStepper/DendoStepper.cpp index 33126f5..cc64144 100644 --- a/components/DendoStepper/DendoStepper.cpp +++ b/components/DendoStepper/DendoStepper.cpp @@ -132,15 +132,15 @@ esp_err_t DendoStepper::runPos(int32_t relative) if (ctrl.status > IDLE) { //currently moving //ctrl.status = ctrl.status==COAST ? COAST : ACC; //stay at coast otherwise switch to ACC ctrl.stepsRemaining = ctrl.stepsToGo - ctrl.stepCnt; - calc(abs(relative + ctrl.stepsRemaining)); //calculate new velolcity profile for new+remaining steps - ESP_LOGW("DendoStepper", "EXTEND running movement (stepsRemaining: %d + stepsNew: %d)", ctrl.stepsRemaining, relative); + calc(abs(relative) + ctrl.stepsRemaining); //calculate new velolcity profile for new+remaining steps + ESP_LOGW("DendoStepper", "EXTEND running movement (stepsRemaining: %d + stepsNew: %d - current state: %d)", ctrl.stepsRemaining, abs(relative), (int)ctrl.status); ESP_ERROR_CHECK(timer_set_alarm_value(conf.timer_group, conf.timer_idx, ctrl.stepInterval)); // set HW timer alarm to stepinterval } else { //current state is IDLE //ctrl.statusPrev = ctrl.status; //update previous status - ctrl.status = ACC; calc(abs(relative)); // calculate velocity profile + ctrl.status = ACC; ESP_ERROR_CHECK(timer_set_alarm_value(conf.timer_group, conf.timer_idx, ctrl.stepInterval)); // set HW timer alarm to stepinterval ESP_ERROR_CHECK(timer_start(conf.timer_group, conf.timer_idx)); // start the timer } @@ -388,8 +388,16 @@ bool DendoStepper::xISR() void DendoStepper::calc(uint32_t targetSteps) { + //only set initial speed if IDLE + if(ctrl.status == 1){ + ctrl.currentSpeed = 0; + ESP_LOGD("DendoStepper", "calc-start: reset speed to 0 (start from idle) %lf\n", ctrl.currentSpeed); + } + else{ + ESP_LOGD("DendoStepper", "calc start: NOT resetting speed (extend from ACC/DEC/COAST): %lf\n", ctrl.currentSpeed); + } //CUSTOM reset counter if already moving - ctrl.stepCnt = 0; + ctrl.stepCnt = 0; //FIXME bugs when set 0 while ISR reads/runs? mutex //steps from ctrl.speed -> 0: ctrl.decSteps = 0.5 * ctrl.dec * (ctrl.speed / ctrl.dec) * (ctrl.speed / ctrl.dec); @@ -397,6 +405,7 @@ void DendoStepper::calc(uint32_t targetSteps) //ctrl.accSteps = 0.5 * ctrl.acc * (ctrl.speed / ctrl.acc) * (ctrl.speed / ctrl.acc); //steps from ctrl.currentSpeed -> ctrl.speed: ctrl.accSteps = 0.5 * ctrl.acc * (ctrl.speed / ctrl.acc) * (ctrl.speed / ctrl.acc) * (ctrl.speed - ctrl.currentSpeed) / ctrl.speed; + ESP_LOGD("DendoStepper", "accSteps: %d currspeed: %lf, ctrlSpeed: %lf\n", ctrl.accSteps, ctrl.currentSpeed, ctrl.speed); if (targetSteps < (ctrl.decSteps + ctrl.accSteps)) { @@ -415,12 +424,21 @@ void DendoStepper::calc(uint32_t targetSteps) ctrl.accInc = (ctrl.targetSpeed - ctrl.currentSpeed) / (double)ctrl.accSteps; ctrl.decInc = ctrl.targetSpeed / (double)ctrl.decSteps; - ctrl.currentSpeed = ctrl.accInc; + //only set initial speed if IDLE + if(ctrl.status == 1){ + ctrl.currentSpeed = ctrl.accInc; + ESP_LOGD("DendoStepper", "`reset curr speeed to accinc: %lf\n", ctrl.currentSpeed); + ESP_LOGD("DendoStepper", "status=%d setting speed to initial value: %lf\n",ctrl.status, ctrl.currentSpeed); + } + else{ + ESP_LOGD("DendoStepper", "status=%d NOT resetting speed to initial value %lf\n",ctrl.status, ctrl.currentSpeed); + } ctrl.stepInterval = TIMER_F / ctrl.currentSpeed; ctrl.stepsToGo = targetSteps; - printf("CALC: speedNow=%.1f, speedTarget=%.1f, accEnd=%d, coastEnd=%d, accSteps=%d, accInc=%.3f\n", + ESP_LOGD("DendoStepper", "DEBUG: accSteps: %d currspeed: %lf, ctrlSpeed: %lf\n", ctrl.accSteps, ctrl.currentSpeed, ctrl.speed); + ESP_LOGD("DendoStepper", "CALC: speedNow=%.1f, speedTarget=%.1f, accEnd=%d, coastEnd=%d, accSteps=%d, accInc=%.3f\n", ctrl.currentSpeed, ctrl.targetSpeed, ctrl.accEnd, ctrl.coastEnd, ctrl.accSteps, ctrl.accInc); STEP_LOGI("calc", "acc end:%u coastend:%u stepstogo:%u speed:%f acc:%f int: %u", ctrl.accEnd, ctrl.coastEnd, ctrl.stepsToGo, ctrl.speed, ctrl.acc, ctrl.stepInterval); } diff --git a/main/guide-stepper.cpp b/main/guide-stepper.cpp index fb6605b..44e4e60 100644 --- a/main/guide-stepper.cpp +++ b/main/guide-stepper.cpp @@ -33,8 +33,8 @@ extern "C" #define SPEED_MAX 60.0 //mm/s #define SPEED 10 //35, 100, 50 rev -#define ACCEL_MS 1000.0 //ms from 0 to max -#define DECEL_MS 1000.0 //ms from max to 0 +#define ACCEL_MS 800.0 //ms from 0 to max +#define DECEL_MS 800.0 //ms from max to 0 #define STEPPER_STEPS_PER_ROT 1600 #define STEPPER_STEPS_PER_MM STEPPER_STEPS_PER_ROT/4 @@ -219,19 +219,21 @@ void task_stepper_test(void *pvParameter) SW_AUTO_CUT.handle(); if (SW_RESET.risingEdge) { + ESP_LOGI(TAG, "button - stop stepper\n "); buzzer.beep(1, 1000, 100); step.stop(); - ESP_LOGI(TAG, "button - stop stepper\n "); } if (SW_PRESET1.risingEdge) { - buzzer.beep(2, 300, 100); - step.runPosMm(30); ESP_LOGI(TAG, "button - moving right\n "); + buzzer.beep(2, 300, 100); + step.setSpeedMm(SPEED, ACCEL_MS, DECEL_MS); + step.runPosMm(15); } if (SW_PRESET3.risingEdge) { - buzzer.beep(1, 500, 100); - step.runPosMm(-30); ESP_LOGI(TAG, "button - moving left\n "); + buzzer.beep(1, 500, 100); + step.setSpeedMm(SPEED, ACCEL_MS, DECEL_MS); + step.runPosMm(-15); } if (SW_PRESET2.risingEdge) { buzzer.beep(1, 100, 100); diff --git a/main/main.cpp b/main/main.cpp index 3b7494b..af2f72a 100644 --- a/main/main.cpp +++ b/main/main.cpp @@ -87,7 +87,7 @@ extern "C" void app_main() esp_log_level_set("switches-analog", ESP_LOG_WARN); esp_log_level_set("control", ESP_LOG_INFO); esp_log_level_set("stepper", ESP_LOG_DEBUG); - esp_log_level_set("Dendostepper", ESP_LOG_WARN); //stepper lib + esp_log_level_set("Dendostepper", ESP_LOG_DEBUG); //stepper lib esp_log_level_set("calc", ESP_LOG_WARN); //stepper lib #ifdef STEPPER_TEST From 279ac0e07e7578a86a6d13908e472bd4f379a62a Mon Sep 17 00:00:00 2001 From: jonny_l480 Date: Sun, 26 Mar 2023 11:41:46 +0200 Subject: [PATCH 8/8] Add more debug output ISSUES - Add debug output in calc - disable test mode for testing with encoder - set loglevel to DEBUG extend feature currently only works well in testing mode with buttons issues when running with encoder: movement gets extended extremely often due to encoder travel interval and rarely does reach target - compared to trigger with buttons -> while debugging noticed that the current speed gets negative and the xISR gets stuck so moves infinitely or not at all ideas: - rounding issue? - SPEED increment also has to be adjusted or set to 0 as step increment? --- components/DendoStepper/DendoStepper.cpp | 12 ++++++++---- main/config.hpp | 2 +- main/guide-stepper.cpp | 2 +- main/main.cpp | 2 +- 4 files changed, 11 insertions(+), 7 deletions(-) diff --git a/components/DendoStepper/DendoStepper.cpp b/components/DendoStepper/DendoStepper.cpp index cc64144..40fa8ad 100644 --- a/components/DendoStepper/DendoStepper.cpp +++ b/components/DendoStepper/DendoStepper.cpp @@ -174,7 +174,7 @@ esp_err_t DendoStepper::runAbs(uint32_t position) bool newDir = (relativeSteps < 0); // CCW if <0, else set CW if (ctrl.dir != newDir){ //direction differs STEP_LOGE("DendoStepper", "DIRECTION HOT-CHANGE NOT SUPPORTED - Waiting for move to finish..."); - while (getState() > IDLE) vTaskDelay(1); //wait for move to finish + while (getState() > IDLE) vTaskDelay(5); //wait for move to finish } } @@ -401,6 +401,7 @@ void DendoStepper::calc(uint32_t targetSteps) //steps from ctrl.speed -> 0: ctrl.decSteps = 0.5 * ctrl.dec * (ctrl.speed / ctrl.dec) * (ctrl.speed / ctrl.dec); + ESP_LOGD("DendoStepper", "decSteps: %d currspeed: %lf, ctrlSpeed: %lf\n", ctrl.decSteps, ctrl.currentSpeed, ctrl.speed); //steps from 0 -> ctrl.speed: //ctrl.accSteps = 0.5 * ctrl.acc * (ctrl.speed / ctrl.acc) * (ctrl.speed / ctrl.acc); //steps from ctrl.currentSpeed -> ctrl.speed: @@ -437,8 +438,11 @@ void DendoStepper::calc(uint32_t targetSteps) ctrl.stepInterval = TIMER_F / ctrl.currentSpeed; ctrl.stepsToGo = targetSteps; - ESP_LOGD("DendoStepper", "DEBUG: accSteps: %d currspeed: %lf, ctrlSpeed: %lf\n", ctrl.accSteps, ctrl.currentSpeed, ctrl.speed); - ESP_LOGD("DendoStepper", "CALC: speedNow=%.1f, speedTarget=%.1f, accEnd=%d, coastEnd=%d, accSteps=%d, accInc=%.3f\n", + //debug log output + ESP_LOGD("DendoStepper", "accSteps: %d, accInc: %lf, decSteps: %d, decInc: %lf", + ctrl.accSteps, ctrl.accInc, ctrl.decSteps, ctrl.decInc); + ESP_LOGD("DendoStepper", "speedNow=%.1f, speedTarget=%.1f, accEnd=%d, coastEnd=%d, accSteps=%d, accInc=%.3f\n", ctrl.currentSpeed, ctrl.targetSpeed, ctrl.accEnd, ctrl.coastEnd, ctrl.accSteps, ctrl.accInc); - STEP_LOGI("calc", "acc end:%u coastend:%u stepstogo:%u speed:%f acc:%f int: %u", ctrl.accEnd, ctrl.coastEnd, ctrl.stepsToGo, ctrl.speed, ctrl.acc, ctrl.stepInterval); + ESP_LOGD("DendoStepper", "acc end:%u coastend:%u stepstogo:%u speed:%f acc:%f int: %u", + ctrl.accEnd, ctrl.coastEnd, ctrl.stepsToGo, ctrl.speed, ctrl.acc, ctrl.stepInterval); } diff --git a/main/config.hpp b/main/config.hpp index 95ca28b..dcd474b 100644 --- a/main/config.hpp +++ b/main/config.hpp @@ -85,7 +85,7 @@ extern "C" { //----- stepper config ----- //-------------------------- //enable stepper test mode (dont start control and encoder task) -#define STEPPER_TEST +//#define STEPPER_TEST #define STEPPER_STEP_PIN GPIO_NUM_18 //mos1 #define STEPPER_DIR_PIN GPIO_NUM_16 //ST3 #define STEPPER_EN_PIN GPIO_NUM_0 //not connected (-> stepper always on) diff --git a/main/guide-stepper.cpp b/main/guide-stepper.cpp index 44e4e60..f798d94 100644 --- a/main/guide-stepper.cpp +++ b/main/guide-stepper.cpp @@ -34,7 +34,7 @@ extern "C" #define SPEED 10 //35, 100, 50 rev #define ACCEL_MS 800.0 //ms from 0 to max -#define DECEL_MS 800.0 //ms from max to 0 +#define DECEL_MS 500.0 //ms from max to 0 #define STEPPER_STEPS_PER_ROT 1600 #define STEPPER_STEPS_PER_MM STEPPER_STEPS_PER_ROT/4 diff --git a/main/main.cpp b/main/main.cpp index af2f72a..18cdc5b 100644 --- a/main/main.cpp +++ b/main/main.cpp @@ -87,7 +87,7 @@ extern "C" void app_main() esp_log_level_set("switches-analog", ESP_LOG_WARN); esp_log_level_set("control", ESP_LOG_INFO); esp_log_level_set("stepper", ESP_LOG_DEBUG); - esp_log_level_set("Dendostepper", ESP_LOG_DEBUG); //stepper lib + esp_log_level_set("DendoStepper", ESP_LOG_DEBUG); //stepper lib esp_log_level_set("calc", ESP_LOG_WARN); //stepper lib #ifdef STEPPER_TEST