stepper: Improve performance of scheduled unsteps
On faster MCUs where a delay is needed between step and unstep use a "busy loop" in the scheduler instead of trying to schedule to the unstep time. This reduces the chance of jitter in the scheduler accumulating. Signed-off-by: Kevin O'Connor <kevin@koconnor.net>
This commit is contained in:
parent
18f4d343f5
commit
f8b0c884b0
|
@ -35,9 +35,10 @@ struct stepper {
|
||||||
#if CONFIG_NO_UNSTEP_DELAY
|
#if CONFIG_NO_UNSTEP_DELAY
|
||||||
uint16_t count;
|
uint16_t count;
|
||||||
#define next_step_time time.waketime
|
#define next_step_time time.waketime
|
||||||
|
#define min_step_time time.waketime
|
||||||
#else
|
#else
|
||||||
uint32_t count;
|
uint32_t count;
|
||||||
uint32_t next_step_time;
|
uint32_t next_step_time, min_step_time;
|
||||||
#endif
|
#endif
|
||||||
struct gpio_out step_pin, dir_pin;
|
struct gpio_out step_pin, dir_pin;
|
||||||
uint32_t position;
|
uint32_t position;
|
||||||
|
@ -54,7 +55,7 @@ enum { SF_LAST_DIR=1<<0, SF_NEXT_DIR=1<<1, SF_INVERT_STEP=1<<2, SF_HAVE_ADD=1<<3
|
||||||
|
|
||||||
// Setup a stepper for the next move in its queue
|
// Setup a stepper for the next move in its queue
|
||||||
static uint_fast8_t
|
static uint_fast8_t
|
||||||
stepper_load_next(struct stepper *s, uint32_t min_next_time)
|
stepper_load_next(struct stepper *s)
|
||||||
{
|
{
|
||||||
struct stepper_move *m = s->first;
|
struct stepper_move *m = s->first;
|
||||||
if (!m) {
|
if (!m) {
|
||||||
|
@ -66,6 +67,7 @@ stepper_load_next(struct stepper *s, uint32_t min_next_time)
|
||||||
}
|
}
|
||||||
|
|
||||||
s->next_step_time += m->interval;
|
s->next_step_time += m->interval;
|
||||||
|
s->time.waketime = s->next_step_time;
|
||||||
s->add = m->add;
|
s->add = m->add;
|
||||||
s->interval = m->interval + m->add;
|
s->interval = m->interval + m->add;
|
||||||
if (CONFIG_NO_UNSTEP_DELAY) {
|
if (CONFIG_NO_UNSTEP_DELAY) {
|
||||||
|
@ -74,17 +76,11 @@ stepper_load_next(struct stepper *s, uint32_t min_next_time)
|
||||||
s->count = m->count;
|
s->count = m->count;
|
||||||
} else {
|
} else {
|
||||||
// On faster mcus, it is necessary to schedule unstep events
|
// On faster mcus, it is necessary to schedule unstep events
|
||||||
// and so there are twice as many events. Also check that the
|
// and so there are twice as many events.
|
||||||
// next step event isn't too close to the last unstep.
|
|
||||||
if (unlikely(timer_is_before(s->next_step_time, min_next_time))) {
|
|
||||||
if ((int32_t)(s->next_step_time - min_next_time)
|
|
||||||
< (int32_t)(-timer_from_us(1000)))
|
|
||||||
shutdown("stepper too far in past");
|
|
||||||
s->time.waketime = min_next_time;
|
|
||||||
} else {
|
|
||||||
s->time.waketime = s->next_step_time;
|
|
||||||
}
|
|
||||||
s->count = m->count * 2;
|
s->count = m->count * 2;
|
||||||
|
if (timer_is_before(s->next_step_time + timer_from_us(1000)
|
||||||
|
, s->min_step_time))
|
||||||
|
shutdown("stepper too far in past");
|
||||||
}
|
}
|
||||||
if (m->flags & MF_DIR) {
|
if (m->flags & MF_DIR) {
|
||||||
s->position = -s->position + m->count;
|
s->position = -s->position + m->count;
|
||||||
|
@ -118,30 +114,32 @@ stepper_event(struct timer *t)
|
||||||
s->interval += s->add;
|
s->interval += s->add;
|
||||||
return SF_RESCHEDULE;
|
return SF_RESCHEDULE;
|
||||||
}
|
}
|
||||||
uint_fast8_t ret = stepper_load_next(s, 0);
|
uint_fast8_t ret = stepper_load_next(s);
|
||||||
gpio_out_toggle(s->step_pin);
|
gpio_out_toggle(s->step_pin);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
// On faster mcus, it is necessary to schedule the unstep event
|
// On faster mcus, it is necessary to wait between step and unstep
|
||||||
uint32_t min_next_time = timer_read_time() + UNSTEP_TIME;
|
uint32_t curtime = timer_read_time();
|
||||||
|
if (unlikely(timer_is_before(curtime, s->min_step_time)))
|
||||||
|
// Can't step/unstep yet - busy wait in scheduler
|
||||||
|
goto busy_wait;
|
||||||
gpio_out_toggle(s->step_pin);
|
gpio_out_toggle(s->step_pin);
|
||||||
|
s->min_step_time = curtime + UNSTEP_TIME;
|
||||||
s->count--;
|
s->count--;
|
||||||
if (likely(s->count & 1))
|
if (s->count & 1)
|
||||||
// Schedule unstep event
|
// Started step - now busy wait in scheduler until unstep ready
|
||||||
goto reschedule_min;
|
goto busy_wait;
|
||||||
if (likely(s->count)) {
|
if (likely(s->count)) {
|
||||||
s->next_step_time += s->interval;
|
s->next_step_time += s->interval;
|
||||||
s->interval += s->add;
|
s->interval += s->add;
|
||||||
if (unlikely(timer_is_before(s->next_step_time, min_next_time)))
|
|
||||||
// The next step event is too close - push it back
|
|
||||||
goto reschedule_min;
|
|
||||||
s->time.waketime = s->next_step_time;
|
s->time.waketime = s->next_step_time;
|
||||||
return SF_RESCHEDULE;
|
return SF_RESCHEDULE;
|
||||||
}
|
}
|
||||||
return stepper_load_next(s, min_next_time);
|
return stepper_load_next(s);
|
||||||
reschedule_min:
|
|
||||||
s->time.waketime = min_next_time;
|
busy_wait:
|
||||||
|
s->time.waketime = curtime;
|
||||||
return SF_RESCHEDULE;
|
return SF_RESCHEDULE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -202,7 +200,9 @@ command_queue_step(uint32_t *args)
|
||||||
s->plast = &m->next;
|
s->plast = &m->next;
|
||||||
} else {
|
} else {
|
||||||
s->first = m;
|
s->first = m;
|
||||||
stepper_load_next(s, s->next_step_time + m->interval);
|
if (!CONFIG_NO_UNSTEP_DELAY)
|
||||||
|
s->min_step_time = s->next_step_time + m->interval;
|
||||||
|
stepper_load_next(s);
|
||||||
sched_add_timer(&s->time);
|
sched_add_timer(&s->time);
|
||||||
}
|
}
|
||||||
irq_enable();
|
irq_enable();
|
||||||
|
@ -232,6 +232,7 @@ command_reset_step_clock(uint32_t *args)
|
||||||
if (s->count)
|
if (s->count)
|
||||||
shutdown("Can't reset time when stepper active");
|
shutdown("Can't reset time when stepper active");
|
||||||
s->next_step_time = waketime;
|
s->next_step_time = waketime;
|
||||||
|
s->min_step_time = s->next_step_time;
|
||||||
s->flags |= SF_LAST_RESET;
|
s->flags |= SF_LAST_RESET;
|
||||||
irq_enable();
|
irq_enable();
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue