Simplify code, Add actual units: TargetPosMm, SpeedMm
- add functions setSpeed (mm/s) setTargetPosMm - add macros and function to define and convert variables to actual units speed in mm/s instead of unknown abs pos in mm instead of steps - simplyfy isr code
This commit is contained in:
parent
63f0da25f1
commit
e8e1070bd1
@ -187,15 +187,23 @@ void task_stepper_test(void *pvParameter)
|
|||||||
if (SW_RESET.risingEdge) {
|
if (SW_RESET.risingEdge) {
|
||||||
switch (state){
|
switch (state){
|
||||||
case 0:
|
case 0:
|
||||||
stepper_setTargetSteps(1000);
|
stepper_setTargetPosMm(50);
|
||||||
|
//stepper_setTargetPosSteps(1000);
|
||||||
state++;
|
state++;
|
||||||
break;
|
break;
|
||||||
case 1:
|
case 1:
|
||||||
stepper_setTargetSteps(100);
|
stepper_setTargetPosMm(80);
|
||||||
|
//stepper_setTargetPosSteps(100);
|
||||||
state++;
|
state++;
|
||||||
break;
|
break;
|
||||||
case 2:
|
case 2:
|
||||||
stepper_setTargetSteps(2000);
|
stepper_setTargetPosMm(20);
|
||||||
|
//stepper_setTargetPosSteps(100);
|
||||||
|
state++;
|
||||||
|
break;
|
||||||
|
case 3:
|
||||||
|
stepper_setTargetPosMm(60);
|
||||||
|
//stepper_setTargetPosSteps(2000);
|
||||||
state = 0;
|
state = 0;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
145
main/stepper.cpp
145
main/stepper.cpp
@ -14,6 +14,13 @@ extern "C" {
|
|||||||
//#define STEPPER_STEP_PIN GPIO_NUM_18 //mos1
|
//#define STEPPER_STEP_PIN GPIO_NUM_18 //mos1
|
||||||
//#define STEPPER_DIR_PIN GPIO_NUM_16 //ST3
|
//#define STEPPER_DIR_PIN GPIO_NUM_16 //ST3
|
||||||
|
|
||||||
|
#define STEPPER_STEPS_PER_MM 200/2 //steps/mm
|
||||||
|
#define STEPPER_SPEED_DEFAULT 20 //mm/s
|
||||||
|
#define STEPPER_SPEED_MIN 4 //mm/s - speed at which stepper immediately starts/stops
|
||||||
|
#define STEPPER_ACCEL_INC 3 //steps/s per cycle
|
||||||
|
#define STEPPER_DECEL_INC 8 //steps/s per cycle
|
||||||
|
|
||||||
|
|
||||||
#define TIMER_F 1000000ULL
|
#define TIMER_F 1000000ULL
|
||||||
#define TICK_PER_S TIMER_S
|
#define TICK_PER_S TIMER_S
|
||||||
#define NS_TO_T_TICKS(x) (x)
|
#define NS_TO_T_TICKS(x) (x)
|
||||||
@ -33,13 +40,17 @@ bool timer_isr(void *arg);
|
|||||||
static timer_group_t timerGroup = TIMER_GROUP_0;
|
static timer_group_t timerGroup = TIMER_GROUP_0;
|
||||||
static timer_idx_t timerIdx = TIMER_0;
|
static timer_idx_t timerIdx = TIMER_0;
|
||||||
|
|
||||||
//move to isr
|
//TODO the below variables can be moved to isr function once debug output is no longer needed
|
||||||
static uint64_t posTarget = 0;
|
static uint64_t posTarget = 0;
|
||||||
static uint64_t posNow = 0;
|
static uint64_t posNow = 0;
|
||||||
static uint64_t stepsToGo = 0;
|
static uint64_t stepsToGo = 0;
|
||||||
static uint32_t speedMin = 20000;
|
static uint32_t speedMin = STEPPER_SPEED_MIN * STEPPER_STEPS_PER_MM;
|
||||||
static uint32_t speedNow = speedMin;
|
static uint32_t speedNow = speedMin;
|
||||||
static int debug = 0;
|
static int debug = 0;
|
||||||
|
static uint32_t speedTarget = STEPPER_SPEED_DEFAULT * STEPPER_STEPS_PER_MM;
|
||||||
|
//TODO/NOTE increment actually has to be re-calculated every run to have linear accel (because also gets called faster/slower)
|
||||||
|
static uint32_t decel_increment = STEPPER_DECEL_INC;
|
||||||
|
static uint32_t accel_increment = STEPPER_ACCEL_INC;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@ -94,7 +105,9 @@ void stepperSw_setTargetSteps(uint64_t target){
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
//======================
|
||||||
|
//===== DEBUG task =====
|
||||||
|
//======================
|
||||||
void task_stepper_debug(void *pvParameter){
|
void task_stepper_debug(void *pvParameter){
|
||||||
while (1){
|
while (1){
|
||||||
ESP_LOGI("stepper-DEBUG",
|
ESP_LOGI("stepper-DEBUG",
|
||||||
@ -105,6 +118,7 @@ void task_stepper_debug(void *pvParameter){
|
|||||||
"posNow=%llu "
|
"posNow=%llu "
|
||||||
"stepsToGo=%llu "
|
"stepsToGo=%llu "
|
||||||
"speedNow=%u "
|
"speedNow=%u "
|
||||||
|
"speedTarget=%u "
|
||||||
"debug=%d ",
|
"debug=%d ",
|
||||||
|
|
||||||
timerIsRunning,
|
timerIsRunning,
|
||||||
@ -114,6 +128,7 @@ void task_stepper_debug(void *pvParameter){
|
|||||||
posNow,
|
posNow,
|
||||||
stepsToGo,
|
stepsToGo,
|
||||||
speedNow,
|
speedNow,
|
||||||
|
speedTarget,
|
||||||
debug
|
debug
|
||||||
);
|
);
|
||||||
|
|
||||||
@ -122,22 +137,24 @@ void task_stepper_debug(void *pvParameter){
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//=====================
|
||||||
|
//===== set speed =====
|
||||||
|
//=====================
|
||||||
|
void stepper_setSpeed(uint32_t speedMmPerS) {
|
||||||
|
ESP_LOGW(TAG, "set target speed from %u to %u mm/s (%u steps/s)",
|
||||||
|
speedTarget, speedMmPerS, speedMmPerS * STEPPER_STEPS_PER_MM);
|
||||||
|
speedTarget = speedMmPerS * STEPPER_STEPS_PER_MM;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
//========================
|
//==========================
|
||||||
//==== set target pos ====
|
//== set target pos STEPS ==
|
||||||
//========================
|
//==========================
|
||||||
void stepper_setTargetSteps(int target_steps) {
|
void stepper_setTargetPosSteps(uint64_t target_steps) {
|
||||||
ESP_LOGW(TAG, "set target steps from %lld to %d (stepsNow: %llu", (long long int)posTarget, target_steps, (long long int)posNow);
|
ESP_LOGW(TAG, "update target position from %llu to %llu steps (stepsNow: %llu", posTarget, target_steps, posNow);
|
||||||
posTarget = target_steps;
|
posTarget = target_steps;
|
||||||
//TODO switch dir pin in isr? not in sync with count
|
|
||||||
//TODO switch direction using negative values as below
|
|
||||||
|
|
||||||
// Update the targetSteps value
|
|
||||||
//ctrl.targetSteps = abs(target_steps);
|
|
||||||
|
|
||||||
// Check if the timer is currently paused
|
// Check if the timer is currently paused
|
||||||
ESP_LOGW(TAG, "check if timer is running %d", timerIsRunning);
|
|
||||||
if (!timerIsRunning){
|
if (!timerIsRunning){
|
||||||
// If the timer is paused, start it again with the updated targetSteps
|
// If the timer is paused, start it again with the updated targetSteps
|
||||||
timerIsRunning = true;
|
timerIsRunning = true;
|
||||||
@ -145,11 +162,17 @@ void stepper_setTargetSteps(int target_steps) {
|
|||||||
ESP_ERROR_CHECK(timer_set_alarm_value(timerGroup, timerIdx, 1000));
|
ESP_ERROR_CHECK(timer_set_alarm_value(timerGroup, timerIdx, 1000));
|
||||||
//timer_set_counter_value(timerGroup, timerIdx, 1000);
|
//timer_set_counter_value(timerGroup, timerIdx, 1000);
|
||||||
ESP_ERROR_CHECK(timer_start(timerGroup, timerIdx));
|
ESP_ERROR_CHECK(timer_start(timerGroup, timerIdx));
|
||||||
ESP_LOGW(TAG, "STARTED TIMER");
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//=========================
|
||||||
|
//=== set target pos MM ===
|
||||||
|
//=========================
|
||||||
|
void stepper_setTargetPosMm(uint32_t posMm){
|
||||||
|
ESP_LOGW(TAG, "set target position to %u mm", posMm);
|
||||||
|
stepper_setTargetPosSteps(posMm * STEPPER_STEPS_PER_MM);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@ -184,86 +207,80 @@ void stepper_init(){
|
|||||||
//=== timer interrupt function ===
|
//=== timer interrupt function ===
|
||||||
//================================
|
//================================
|
||||||
bool timer_isr(void *arg) {
|
bool timer_isr(void *arg) {
|
||||||
// Generate pulse for stepper motor
|
|
||||||
//turn pin on (fast)
|
|
||||||
GPIO.out_w1ts = (1ULL << STEPPER_STEP_PIN);
|
|
||||||
|
|
||||||
|
|
||||||
|
//-----------------
|
||||||
//--- variables ---
|
//--- variables ---
|
||||||
static uint32_t speedTarget = 100000;
|
//-----------------
|
||||||
//FIXME increment actually has to be re-calculated every run to have linear accel (because also gets called faster/slower)
|
//TODO used (currently global) variables here
|
||||||
static uint32_t decel_increment = 200;
|
|
||||||
static uint32_t accel_increment = 150;
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
//-----------------------------------
|
||||||
//--- define direction, stepsToGo ---
|
//--- define direction, stepsToGo ---
|
||||||
//int64_t delta = (int)posTarget - (int)posNow;
|
//-----------------------------------
|
||||||
//bool directionTarget = delta >= 0 ? 1 : 0;
|
//Note: the idea is that the stepper has to decelerate to min speed first before changeing the direction
|
||||||
if (posTarget >= posNow) {
|
//define target direction depending on position difference
|
||||||
directionTarget = 1;
|
bool directionTarget = posTarget > posNow ? 1 : 0;
|
||||||
} else {
|
//DIRECTION DIFFERS (change)
|
||||||
directionTarget = 0;
|
|
||||||
}
|
|
||||||
//directionTarget = 1;
|
|
||||||
//direction = 1;
|
|
||||||
//gpio_set_level(STEPPER_DIR_PIN, direction);
|
|
||||||
if ( (direction != directionTarget) && (posTarget != posNow)) {
|
if ( (direction != directionTarget) && (posTarget != posNow)) {
|
||||||
//ESP_LOGW(TAG, "direction differs! new: %d", direction);
|
if (stepsToGo == 0){ //standstill
|
||||||
if (stepsToGo == 0){
|
direction = directionTarget; //switch direction
|
||||||
direction = directionTarget; //switch direction if almost idle
|
|
||||||
gpio_set_level(STEPPER_DIR_PIN, direction);
|
gpio_set_level(STEPPER_DIR_PIN, direction);
|
||||||
//stepsToGo = abs((int64_t)posTarget - (int64_t)posNow);
|
stepsToGo = abs(int64_t(posTarget - posNow));
|
||||||
stepsToGo = 2;
|
|
||||||
} else {
|
} else {
|
||||||
//stepsToGo = speedNow / decel_increment; //set steps to decel to min speed
|
|
||||||
//set to minimun decel steps
|
//set to minimun decel steps
|
||||||
stepsToGo = (speedNow - speedMin) / decel_increment;
|
stepsToGo = (speedNow - speedMin) / decel_increment;
|
||||||
}
|
}
|
||||||
} else if (direction == 1) {
|
|
||||||
stepsToGo = posTarget - posNow;
|
|
||||||
//stepsToGo = abs((int64_t)posTarget - (int64_t)posNow);
|
|
||||||
} else {
|
|
||||||
stepsToGo = posNow - posTarget;
|
|
||||||
//stepsToGo = abs((int64_t)posTarget - (int64_t)posNow);
|
|
||||||
}
|
}
|
||||||
//TODO fix direction code above currently ony works with the below line instead
|
//NORMAL (any direction 0/1)
|
||||||
//stepsToGo = abs((int64_t)posTarget - (int64_t)posNow);
|
else {
|
||||||
|
stepsToGo = abs(int64_t(posTarget - posNow));
|
||||||
|
}
|
||||||
|
|
||||||
|
//--------------------
|
||||||
//--- define speed ---
|
//--- define speed ---
|
||||||
|
//--------------------
|
||||||
|
//FIXME noticed crash: division by 0 when min speed > target speed
|
||||||
uint64_t stepsDecelRemaining = (speedNow - speedMin) / decel_increment;
|
uint64_t stepsDecelRemaining = (speedNow - speedMin) / decel_increment;
|
||||||
|
//DECELERATE
|
||||||
if (stepsToGo <= stepsDecelRemaining) {
|
if (stepsToGo <= stepsDecelRemaining) {
|
||||||
|
//FIXME if stepsToGo gets updated (lowered) close to target while close to target, the stepper may stop too fast -> implement possibility to 'overshoot and reverse'?
|
||||||
if ((speedNow - speedMin) > decel_increment) {
|
if ((speedNow - speedMin) > decel_increment) {
|
||||||
speedNow -= decel_increment;
|
speedNow -= decel_increment;
|
||||||
} else {
|
} else {
|
||||||
speedNow = speedMin; //PAUSE HERE??? / irrelevant?
|
speedNow = speedMin; //PAUSE HERE??? / irrelevant?
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
//ACCELERATE
|
||||||
else if (speedNow < speedTarget) {
|
else if (speedNow < speedTarget) {
|
||||||
speedNow += accel_increment;
|
speedNow += accel_increment;
|
||||||
if (speedNow > speedTarget) speedNow = speedTarget;
|
if (speedNow > speedTarget) speedNow = speedTarget;
|
||||||
}
|
}
|
||||||
|
//COASTING
|
||||||
else { //not relevant?
|
else { //not relevant?
|
||||||
speedNow = speedTarget;
|
speedNow = speedTarget;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//-------------------------------
|
||||||
//--- update timer ---
|
//--- update timer, increment ---
|
||||||
|
//-------------------------------
|
||||||
|
//AT TARGET -> STOP
|
||||||
if (stepsToGo == 0) {
|
if (stepsToGo == 0) {
|
||||||
timer_pause(timerGroup, timerIdx);
|
timer_pause(timerGroup, timerIdx);
|
||||||
timerIsRunning = false;
|
timerIsRunning = false;
|
||||||
}
|
speedNow = speedMin;
|
||||||
else {
|
return 1;
|
||||||
ESP_ERROR_CHECK(timer_set_alarm_value(timerGroup, timerIdx, TIMER_BASE_CLK / speedNow));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//STEPS REMAINING -> NEXT STEP
|
||||||
|
//update timer with new speed
|
||||||
|
ESP_ERROR_CHECK(timer_set_alarm_value(timerGroup, timerIdx, TIMER_F / speedNow));
|
||||||
|
|
||||||
//--- increment position ---
|
//generate pulse
|
||||||
if (stepsToGo > 0){
|
GPIO.out_w1ts = (1ULL << STEPPER_STEP_PIN); //turn on (fast)
|
||||||
stepsToGo --; //TODO increment at start, check at start??
|
ets_delay_us(10);
|
||||||
}
|
GPIO.out_w1tc = (1ULL << STEPPER_STEP_PIN); //turn off (fast)
|
||||||
|
|
||||||
|
//increment pos
|
||||||
|
stepsToGo --;
|
||||||
if (direction == 1){
|
if (direction == 1){
|
||||||
posNow ++;
|
posNow ++;
|
||||||
} else {
|
} else {
|
||||||
@ -271,15 +288,9 @@ bool timer_isr(void *arg) {
|
|||||||
if (posNow != 0){
|
if (posNow != 0){
|
||||||
posNow --;
|
posNow --;
|
||||||
} else {
|
} else {
|
||||||
//ERR posNow would be negative?
|
ESP_LOGE(TAG,"isr: posNow would be negative - ignoring decrement");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// Generate pulse for stepper motor
|
|
||||||
//turn pin off (fast)
|
|
||||||
GPIO.out_w1tc = (1ULL << STEPPER_STEP_PIN);
|
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2,7 +2,10 @@
|
|||||||
|
|
||||||
//init stepper pins and timer
|
//init stepper pins and timer
|
||||||
void stepper_init();
|
void stepper_init();
|
||||||
void stepper_setTargetSteps(int steps);
|
//set absolute target position in steps
|
||||||
|
void stepper_setTargetPosSteps(uint64_t steps);
|
||||||
|
//set absolute target position in millimeters
|
||||||
|
void stepper_setTargetPosMm(uint32_t posMm);
|
||||||
|
|
||||||
//task that periodically logs variables for debugging stepper driver
|
//task that periodically logs variables for debugging stepper driver
|
||||||
void task_stepper_debug(void *pvParameter);
|
void task_stepper_debug(void *pvParameter);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user