AP_HAL_ChibiOS: add support for ProfiLEDs

This commit is contained in:
Peter Hall 2020-02-22 23:55:47 +00:00 committed by Andrew Tridgell
parent 53ec370103
commit 07dc4096fa
2 changed files with 182 additions and 45 deletions

View File

@ -399,7 +399,7 @@ void RCOutput::push_local(void)
pwmEnableChannel(group.pwm_drv, j, width); pwmEnableChannel(group.pwm_drv, j, width);
} }
#ifndef DISABLE_DSHOT #ifndef DISABLE_DSHOT
else if (is_dshot_protocol(group.current_mode) || group.current_mode == MODE_NEOPIXEL) { else if (is_dshot_protocol(group.current_mode) || group.current_mode == MODE_NEOPIXEL || group.current_mode == MODE_PROFILED) {
// set period_us to time for pulse output, to enable very fast rates // set period_us to time for pulse output, to enable very fast rates
period_us = dshot_pulse_time_us; period_us = dshot_pulse_time_us;
} }
@ -410,6 +410,7 @@ void RCOutput::push_local(void)
if (group.current_mode == MODE_PWM_ONESHOT || if (group.current_mode == MODE_PWM_ONESHOT ||
group.current_mode == MODE_PWM_ONESHOT125 || group.current_mode == MODE_PWM_ONESHOT125 ||
group.current_mode == MODE_NEOPIXEL || group.current_mode == MODE_NEOPIXEL ||
group.current_mode == MODE_PROFILED ||
is_dshot_protocol(group.current_mode)) { is_dshot_protocol(group.current_mode)) {
need_trigger |= (1U<<i); need_trigger |= (1U<<i);
} }
@ -488,7 +489,7 @@ bool RCOutput::mode_requires_dma(enum output_mode mode) const
#ifdef DISABLE_DSHOT #ifdef DISABLE_DSHOT
return false; return false;
#else #else
return is_dshot_protocol(mode) || (mode == MODE_NEOPIXEL); return is_dshot_protocol(mode) || (mode == MODE_NEOPIXEL) || (mode == MODE_PROFILED);
#endif //#ifdef DISABLE_DSHOT #endif //#ifdef DISABLE_DSHOT
} }
@ -618,24 +619,32 @@ void RCOutput::set_group_mode(pwm_group &group)
break; break;
case MODE_NEOPIXEL: case MODE_NEOPIXEL:
case MODE_PROFILED:
{ {
uint8_t bits_per_pixel = 24;
bool active_high = true;
if (group.current_mode == MODE_PROFILED) {
bits_per_pixel = 25;
active_high = false;
}
const uint32_t rate = protocol_bitrate(group.current_mode); const uint32_t rate = protocol_bitrate(group.current_mode);
const uint32_t bit_period = 20; const uint32_t bit_period = 20;
// configure timer driver for DMAR at requested rate // configure timer driver for DMAR at requested rate
const uint8_t pad_end_bits = 8; const uint8_t pad_end_bits = 8;
const uint8_t pad_start_bits = 1; const uint8_t pad_start_bits = 1;
const uint8_t bits_per_pixel = 24;
const uint8_t channels_per_group = 4; const uint8_t channels_per_group = 4;
const uint16_t neopixel_bit_length = bits_per_pixel * channels_per_group * group.neopixel_nleds + (pad_start_bits + pad_end_bits) * channels_per_group; const uint16_t bit_length = bits_per_pixel * channels_per_group * group.serial_nleds + (pad_start_bits + pad_end_bits) * channels_per_group;
const uint16_t neopixel_buffer_length = neopixel_bit_length * sizeof(uint32_t); const uint16_t buffer_length = bit_length * sizeof(uint32_t);
if (!setup_group_DMA(group, rate, bit_period, true, neopixel_buffer_length, true)) { if (!setup_group_DMA(group, rate, bit_period, active_high, buffer_length, true)) {
group.current_mode = MODE_PWM_NONE; group.current_mode = MODE_PWM_NONE;
break; break;
} }
// calculate min time between pulses // calculate min time between pulses
dshot_pulse_time_us = 1000000UL * neopixel_bit_length / rate; dshot_pulse_time_us = 1000000UL * bit_length / rate;
break; break;
} }
@ -901,12 +910,14 @@ void RCOutput::timer_tick(void)
dshot_send(group, true); dshot_send(group, true);
} }
} }
if (neopixel_pending && chMtxTryLock(&trigger_mutex)) { if (serial_led_pending && chMtxTryLock(&trigger_mutex)) {
neopixel_pending = false; serial_led_pending = false;
for (uint8_t j = 0; j < NUM_GROUPS; j++) { for (uint8_t j = 0; j < NUM_GROUPS; j++) {
pwm_group &group = pwm_group_list[j]; pwm_group &group = pwm_group_list[j];
if (group.current_mode == MODE_NEOPIXEL) { if (group.serial_led_pending && (group.current_mode == MODE_NEOPIXEL || group.current_mode == MODE_PROFILED)) {
neopixel_send(group); group.serial_led_pending = !serial_led_send(group);
group.prepared_send = group.serial_led_pending;
serial_led_pending |= group.serial_led_pending;
} }
} }
chMtxUnlock(&trigger_mutex); chMtxUnlock(&trigger_mutex);
@ -1081,14 +1092,15 @@ void RCOutput::dshot_send(pwm_group &group, bool blocking)
/* /*
send a set of Neopixel packets for a channel group send a set of Serial LED packets for a channel group
return true if send was successful
*/ */
void RCOutput::neopixel_send(pwm_group &group) bool RCOutput::serial_led_send(pwm_group &group)
{ {
#ifndef DISABLE_DSHOT #ifndef DISABLE_DSHOT
if (irq.waiter || !group.dma_handle->lock_nonblock()) { if (irq.waiter || !group.dma_handle->lock_nonblock()) {
// doing serial output, don't send Neopixel pulses // doing serial output, don't send Serial LED pulses
return; return false;
} }
// start sending the pulses out // start sending the pulses out
@ -1096,6 +1108,7 @@ void RCOutput::neopixel_send(pwm_group &group)
group.last_dmar_send_us = AP_HAL::micros64(); group.last_dmar_send_us = AP_HAL::micros64();
#endif //#ifndef DISABLE_DSHOT #endif //#ifndef DISABLE_DSHOT
return true;
} }
@ -1666,6 +1679,8 @@ uint32_t RCOutput::protocol_bitrate(const enum output_mode mode) const
return 1200000; return 1200000;
case MODE_NEOPIXEL: case MODE_NEOPIXEL:
return 800000; return 800000;
case MODE_PROFILED:
return 1500000; // experiment winding this up 3000000 max from data sheet
default: default:
// use 1 to prevent a possible divide-by-zero // use 1 to prevent a possible divide-by-zero
return 1; return 1;
@ -1673,23 +1688,44 @@ uint32_t RCOutput::protocol_bitrate(const enum output_mode mode) const
} }
/* /*
setup neopixel (WS2812B) output for a given channel number, with setup serial led output for a given channel number, with
the given max number of LEDs in the chain. the given max number of LEDs in the chain.
*/ */
bool RCOutput::set_neopixel_num_LEDs(const uint16_t chan, uint8_t num_leds) bool RCOutput::set_serial_led_num_LEDs(const uint16_t chan, uint8_t num_leds, output_mode mode, uint16_t clock_mask)
{ {
uint8_t i; uint8_t i;
pwm_group *grp = find_chan(chan, i); pwm_group *grp = find_chan(chan, i);
if (!grp) { if (!grp) {
return false; return false;
} }
if (grp->neopixel_nleds == num_leds && grp->current_mode == MODE_NEOPIXEL) {
// no change switch (mode) {
return true; case MODE_NEOPIXEL: {
grp->serial_nleds = MAX(num_leds, grp->serial_nleds);
break;
}
case MODE_PROFILED: {
// ProfiLED requires two dummy LED's to mark end of transmission
grp->serial_nleds = MAX(num_leds + 2, grp->serial_nleds);
// Enable any clock channels in the same group
grp->clock_mask = 0;
for (uint8_t j = 0; j < 4; j++) {
if ((clock_mask & (1U<<(grp->chan[j] + chan_offset))) != 0) {
grp->clock_mask |= 1U<<j;
}
}
break;
}
default: {
return false;
}
} }
grp->neopixel_nleds = MAX(num_leds, grp->neopixel_nleds);
set_output_mode(1U<<chan, MODE_NEOPIXEL); set_output_mode(1U<<chan, mode);
return grp->current_mode == MODE_NEOPIXEL;
return grp->current_mode == mode;
} }
/* /*
@ -1712,36 +1748,127 @@ void RCOutput::_set_neopixel_rgb_data(pwm_group *grp, uint8_t idx, uint8_t led,
} }
/* /*
setup neopixel (WS2812B) output data for a given output channel ProfiLED frame for a given output channel
channel is active high and bits inverted to get clock rising edge away from data rising edge
*/
void RCOutput::_set_profiled_rgb_data(pwm_group *grp, uint8_t idx, uint8_t led, uint8_t red, uint8_t green, uint8_t blue)
{
const uint8_t pad_start_bits = 1;
const uint8_t bit_length = 25;
const uint8_t stride = 4;
uint32_t *buf = grp->dma_buffer + (led * bit_length + pad_start_bits) * stride + idx;
uint32_t bits = 0x1000000 | (blue<<16) | (red<<8) | green;
const uint32_t BIT_1 = 14 * grp->bit_width_mul;
for (uint16_t b=0; b < bit_length; b++) {
buf[b * stride] = (bits & 0x1000000) ? 0 : BIT_1;
bits <<= 1;
}
}
/*
ProfiLED blank frame for a given output channel
channel is active high and bits inverted to get clock rising edge away from data rising edge
*/
void RCOutput::_set_profiled_blank_frame(pwm_group *grp, uint8_t idx, uint8_t led)
{
const uint8_t pad_start_bits = 1;
const uint8_t bit_length = 25;
const uint8_t stride = 4;
uint32_t *buf = grp->dma_buffer + (led * bit_length + pad_start_bits) * stride + idx;
const uint32_t BIT_1 = 14 * grp->bit_width_mul;
for (uint16_t b=0; b < bit_length; b++) {
buf[b * stride] = BIT_1;
}
}
/*
setup ProfiLED clock frame for a given output channel
*/
void RCOutput::_set_profiled_clock(pwm_group *grp, uint8_t idx, uint8_t led)
{
const uint8_t pad_start_bits = 1;
const uint8_t bit_length = 25;
const uint8_t stride = 4;
uint32_t *buf = grp->dma_buffer + (led * bit_length + pad_start_bits) * stride + idx;
const uint32_t BIT_1 = 7 * grp->bit_width_mul;
for (uint16_t b=0; b < bit_length; b++) {
buf[b * stride] = BIT_1;
}
}
/*
setup serial LED output data for a given output channel
and a LED number. LED -1 is all LEDs and a LED number. LED -1 is all LEDs
*/ */
void RCOutput::set_neopixel_rgb_data(const uint16_t chan, int8_t led, uint8_t red, uint8_t green, uint8_t blue) void RCOutput::set_serial_led_rgb_data(const uint16_t chan, int8_t led, uint8_t red, uint8_t green, uint8_t blue)
{ {
uint8_t i; uint8_t i;
pwm_group *grp = find_chan(chan, i); pwm_group *grp = find_chan(chan, i);
if (!grp) { if (!grp) {
return; return;
} }
if (led >= grp->serial_nleds || (grp->current_mode != MODE_NEOPIXEL && grp->current_mode != MODE_PROFILED)) {
return;
}
if (led == -1) { if (led == -1) {
for (uint8_t n=0; n<grp->neopixel_nleds; n++) { grp->prepared_send = true;
set_neopixel_rgb_data(chan, n, red, green, blue); for (uint8_t n=0; n<grp->serial_nleds; n++) {
set_serial_led_rgb_data(chan, n, red, green, blue);
} }
return; return;
} else if (!grp->prepared_send) {
// if not ouput clock and trailing frames, run through all LED's to do it now
set_serial_led_rgb_data(chan, -1, 0, 0, 0);
} }
if (led < -1 || led >= grp->neopixel_nleds) { switch (grp->current_mode) {
return; case MODE_NEOPIXEL: {
} _set_neopixel_rgb_data(grp, i, uint8_t(led), red, green, blue);
break;
}
_set_neopixel_rgb_data(grp, i, uint8_t(led), red, green, blue); case MODE_PROFILED: {
if (led < grp->serial_nleds - 2) {
_set_profiled_rgb_data(grp, i, uint8_t(led), red, green, blue);
} else {
_set_profiled_blank_frame(grp, i, uint8_t(led));
}
for (uint8_t j = 0; j < 4; j++) {
if ((grp->clock_mask & 1U<<j) != 0) {
_set_profiled_clock(grp, j, uint8_t(led));
}
}
break;
}
default: {
return;
}
}
} }
/* /*
trigger send of neopixel data trigger send of serial led data for one group
*/ */
void RCOutput::neopixel_send(void) void RCOutput::serial_led_send(const uint16_t chan)
{ {
neopixel_pending = true; uint8_t i;
pwm_group *grp = find_chan(chan, i);
if (!grp) {
return;
}
if (grp->current_mode != MODE_NEOPIXEL && grp->current_mode != MODE_PROFILED) {
return;
}
if (grp->prepared_send) {
serial_led_pending = true;
grp->serial_led_pending = true;
}
} }
#endif // HAL_USE_PWM #endif // HAL_USE_PWM

View File

@ -157,21 +157,21 @@ public:
} }
/* /*
setup neopixel (WS2812B) output for a given channel number, with setup serial LED output for a given channel number, with
the given max number of LEDs in the chain. the given max number of LEDs in the chain.
*/ */
bool set_neopixel_num_LEDs(const uint16_t chan, uint8_t num_leds) override; bool set_serial_led_num_LEDs(const uint16_t chan, uint8_t num_leds, output_mode mode = MODE_PWM_NONE, uint16_t clock_mask = 0) override;
/* /*
setup neopixel (WS2812B) output data for a given output channel setup serial LED output data for a given output channel
and LEDs number. LED -1 is all LEDs and LEDs number. LED -1 is all LEDs
*/ */
void set_neopixel_rgb_data(const uint16_t chan, int8_t led, uint8_t red, uint8_t green, uint8_t blue) override; void set_serial_led_rgb_data(const uint16_t chan, int8_t led, uint8_t red, uint8_t green, uint8_t blue) override;
/* /*
trigger send of neopixel data trigger send of serial LED data
*/ */
void neopixel_send(void) override; void serial_led_send(const uint16_t chan) override;
private: private:
struct pwm_group { struct pwm_group {
@ -201,7 +201,10 @@ private:
bool in_serial_dma; bool in_serial_dma;
uint64_t last_dmar_send_us; uint64_t last_dmar_send_us;
virtual_timer_t dma_timeout; virtual_timer_t dma_timeout;
uint8_t neopixel_nleds; uint8_t serial_nleds;
uint8_t clock_mask;
bool serial_led_pending;
bool prepared_send;
// serial output // serial output
struct { struct {
@ -336,10 +339,11 @@ private:
uint16_t telem_request_mask; uint16_t telem_request_mask;
/* /*
NeoPixel handling. Max of 32 LEDs uses max 12k of memory per group Serial lED handling. Max of 32 LEDs uses max 12k of memory per group
return true if send was successful
*/ */
void neopixel_send(pwm_group &group); bool serial_led_send(pwm_group &group);
bool neopixel_pending; bool serial_led_pending;
void dma_allocate(Shared_DMA *ctx); void dma_allocate(Shared_DMA *ctx);
void dma_deallocate(Shared_DMA *ctx); void dma_deallocate(Shared_DMA *ctx);
@ -357,10 +361,16 @@ private:
/* /*
setup neopixel (WS2812B) output data for a given output channel setup neopixel (WS2812B) output data for a given output channel
and LEDs number. LED -1 is all LEDs
*/ */
void _set_neopixel_rgb_data(pwm_group *grp, uint8_t idx, uint8_t led, uint8_t red, uint8_t green, uint8_t blue); void _set_neopixel_rgb_data(pwm_group *grp, uint8_t idx, uint8_t led, uint8_t red, uint8_t green, uint8_t blue);
/*
setup ProfiLED output data for a given output channel
*/
void _set_profiled_rgb_data(pwm_group *grp, uint8_t idx, uint8_t led, uint8_t red, uint8_t green, uint8_t blue);
void _set_profiled_clock(pwm_group *grp, uint8_t idx, uint8_t led);
void _set_profiled_blank_frame(pwm_group *grp, uint8_t idx, uint8_t led);
// serial output support // serial output support
static const eventmask_t serial_event_mask = EVENT_MASK(1); static const eventmask_t serial_event_mask = EVENT_MASK(1);
bool serial_write_byte(uint8_t b); bool serial_write_byte(uint8_t b);