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) {
|
||||
switch (state){
|
||||
case 0:
|
||||
stepper_setTargetSteps(1000);
|
||||
stepper_setTargetPosMm(50);
|
||||
//stepper_setTargetPosSteps(1000);
|
||||
state++;
|
||||
break;
|
||||
case 1:
|
||||
stepper_setTargetSteps(100);
|
||||
stepper_setTargetPosMm(80);
|
||||
//stepper_setTargetPosSteps(100);
|
||||
state++;
|
||||
break;
|
||||
case 2:
|
||||
stepper_setTargetSteps(2000);
|
||||
stepper_setTargetPosMm(20);
|
||||
//stepper_setTargetPosSteps(100);
|
||||
state++;
|
||||
break;
|
||||
case 3:
|
||||
stepper_setTargetPosMm(60);
|
||||
//stepper_setTargetPosSteps(2000);
|
||||
state = 0;
|
||||
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_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 TICK_PER_S TIMER_S
|
||||
#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_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 posNow = 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 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){
|
||||
while (1){
|
||||
ESP_LOGI("stepper-DEBUG",
|
||||
@ -105,6 +118,7 @@ void task_stepper_debug(void *pvParameter){
|
||||
"posNow=%llu "
|
||||
"stepsToGo=%llu "
|
||||
"speedNow=%u "
|
||||
"speedTarget=%u "
|
||||
"debug=%d ",
|
||||
|
||||
timerIsRunning,
|
||||
@ -114,6 +128,7 @@ void task_stepper_debug(void *pvParameter){
|
||||
posNow,
|
||||
stepsToGo,
|
||||
speedNow,
|
||||
speedTarget,
|
||||
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 ====
|
||||
//========================
|
||||
void stepper_setTargetSteps(int target_steps) {
|
||||
ESP_LOGW(TAG, "set target steps from %lld to %d (stepsNow: %llu", (long long int)posTarget, target_steps, (long long int)posNow);
|
||||
//==========================
|
||||
//== set target pos STEPS ==
|
||||
//==========================
|
||||
void stepper_setTargetPosSteps(uint64_t target_steps) {
|
||||
ESP_LOGW(TAG, "update target position from %llu to %llu steps (stepsNow: %llu", posTarget, target_steps, posNow);
|
||||
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
|
||||
ESP_LOGW(TAG, "check if timer is running %d", timerIsRunning);
|
||||
if (!timerIsRunning){
|
||||
// If the timer is paused, start it again with the updated targetSteps
|
||||
timerIsRunning = true;
|
||||
@ -145,11 +162,17 @@ void stepper_setTargetSteps(int target_steps) {
|
||||
ESP_ERROR_CHECK(timer_set_alarm_value(timerGroup, timerIdx, 1000));
|
||||
//timer_set_counter_value(timerGroup, timerIdx, 1000);
|
||||
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 ===
|
||||
//================================
|
||||
bool timer_isr(void *arg) {
|
||||
// Generate pulse for stepper motor
|
||||
//turn pin on (fast)
|
||||
GPIO.out_w1ts = (1ULL << STEPPER_STEP_PIN);
|
||||
|
||||
|
||||
//-----------------
|
||||
//--- 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)
|
||||
static uint32_t decel_increment = 200;
|
||||
static uint32_t accel_increment = 150;
|
||||
|
||||
|
||||
//-----------------
|
||||
//TODO used (currently global) variables here
|
||||
|
||||
//-----------------------------------
|
||||
//--- define direction, stepsToGo ---
|
||||
//int64_t delta = (int)posTarget - (int)posNow;
|
||||
//bool directionTarget = delta >= 0 ? 1 : 0;
|
||||
if (posTarget >= posNow) {
|
||||
directionTarget = 1;
|
||||
} else {
|
||||
directionTarget = 0;
|
||||
}
|
||||
//directionTarget = 1;
|
||||
//direction = 1;
|
||||
//gpio_set_level(STEPPER_DIR_PIN, direction);
|
||||
//-----------------------------------
|
||||
//Note: the idea is that the stepper has to decelerate to min speed first before changeing the direction
|
||||
//define target direction depending on position difference
|
||||
bool directionTarget = posTarget > posNow ? 1 : 0;
|
||||
//DIRECTION DIFFERS (change)
|
||||
if ( (direction != directionTarget) && (posTarget != posNow)) {
|
||||
//ESP_LOGW(TAG, "direction differs! new: %d", direction);
|
||||
if (stepsToGo == 0){
|
||||
direction = directionTarget; //switch direction if almost idle
|
||||
if (stepsToGo == 0){ //standstill
|
||||
direction = directionTarget; //switch direction
|
||||
gpio_set_level(STEPPER_DIR_PIN, direction);
|
||||
//stepsToGo = abs((int64_t)posTarget - (int64_t)posNow);
|
||||
stepsToGo = 2;
|
||||
stepsToGo = abs(int64_t(posTarget - posNow));
|
||||
} else {
|
||||
//stepsToGo = speedNow / decel_increment; //set steps to decel to min speed
|
||||
//set to minimun decel steps
|
||||
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
|
||||
//stepsToGo = abs((int64_t)posTarget - (int64_t)posNow);
|
||||
|
||||
//NORMAL (any direction 0/1)
|
||||
else {
|
||||
stepsToGo = abs(int64_t(posTarget - posNow));
|
||||
}
|
||||
|
||||
//--------------------
|
||||
//--- define speed ---
|
||||
//--------------------
|
||||
//FIXME noticed crash: division by 0 when min speed > target speed
|
||||
uint64_t stepsDecelRemaining = (speedNow - speedMin) / decel_increment;
|
||||
|
||||
//DECELERATE
|
||||
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) {
|
||||
speedNow -= decel_increment;
|
||||
} else {
|
||||
speedNow = speedMin; //PAUSE HERE??? / irrelevant?
|
||||
}
|
||||
}
|
||||
//ACCELERATE
|
||||
else if (speedNow < speedTarget) {
|
||||
speedNow += accel_increment;
|
||||
if (speedNow > speedTarget) speedNow = speedTarget;
|
||||
}
|
||||
//COASTING
|
||||
else { //not relevant?
|
||||
speedNow = speedTarget;
|
||||
}
|
||||
|
||||
|
||||
//--- update timer ---
|
||||
//-------------------------------
|
||||
//--- update timer, increment ---
|
||||
//-------------------------------
|
||||
//AT TARGET -> STOP
|
||||
if (stepsToGo == 0) {
|
||||
timer_pause(timerGroup, timerIdx);
|
||||
timerIsRunning = false;
|
||||
}
|
||||
else {
|
||||
ESP_ERROR_CHECK(timer_set_alarm_value(timerGroup, timerIdx, TIMER_BASE_CLK / speedNow));
|
||||
speedNow = speedMin;
|
||||
return 1;
|
||||
}
|
||||
|
||||
//STEPS REMAINING -> NEXT STEP
|
||||
//update timer with new speed
|
||||
ESP_ERROR_CHECK(timer_set_alarm_value(timerGroup, timerIdx, TIMER_F / speedNow));
|
||||
|
||||
//--- increment position ---
|
||||
if (stepsToGo > 0){
|
||||
stepsToGo --; //TODO increment at start, check at start??
|
||||
}
|
||||
//generate pulse
|
||||
GPIO.out_w1ts = (1ULL << STEPPER_STEP_PIN); //turn on (fast)
|
||||
ets_delay_us(10);
|
||||
GPIO.out_w1tc = (1ULL << STEPPER_STEP_PIN); //turn off (fast)
|
||||
|
||||
//increment pos
|
||||
stepsToGo --;
|
||||
if (direction == 1){
|
||||
posNow ++;
|
||||
} else {
|
||||
@ -271,15 +288,9 @@ bool timer_isr(void *arg) {
|
||||
if (posNow != 0){
|
||||
posNow --;
|
||||
} 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;
|
||||
}
|
||||
|
||||
|
@ -2,7 +2,10 @@
|
||||
|
||||
//init stepper pins and timer
|
||||
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
|
||||
void task_stepper_debug(void *pvParameter);
|
||||
|
Loading…
x
Reference in New Issue
Block a user