diff --git a/src/config/feature.h b/src/config/feature.h index 2df2bb06b..a7fb04ba9 100644 --- a/src/config/feature.h +++ b/src/config/feature.h @@ -6,6 +6,7 @@ #define USE_BLACKBOX #define USE_MAX7456 +#define USE_RGB_LED #define USE_MOTOR_DSHOT #define USE_MOTOR_PWM diff --git a/src/core/main.c b/src/core/main.c index da3cc9bb9..7b01c0543 100644 --- a/src/core/main.c +++ b/src/core/main.c @@ -13,7 +13,6 @@ #include "driver/fmc.h" #include "driver/gpio.h" #include "driver/motor.h" -#include "driver/reset.h" #include "driver/rgb_led.h" #include "driver/serial.h" #include "driver/spi.h" @@ -96,6 +95,8 @@ __attribute__((__used__)) int main() { led_init(); led_on(LEDALL); + rgb_led_init(); + debug_pin_init(); buzzer_init(); @@ -133,7 +134,6 @@ __attribute__((__used__)) int main() { rx_init(); vtx_init(); - rgb_init(); blackbox_init(); imu_init(); @@ -174,14 +174,7 @@ __attribute__((__used__)) int main() { // handle led commands led_update(); - -#if (RGB_LED_NUMBER > 0) - // RGB led control - rgb_led_lvc(); -#ifdef RGB_LED_DMA - rgb_dma_start(); -#endif -#endif + rgb_led_update(); buzzer_update(); vtx_update(); diff --git a/src/core/target.h b/src/core/target.h index 8aacbf9dd..344de9b60 100644 --- a/src/core/target.h +++ b/src/core/target.h @@ -184,6 +184,7 @@ typedef struct { gpio_pins_t fpv; gpio_pins_t vbat; gpio_pins_t ibat; + gpio_pins_t rgb_led; target_invert_pin_t sdcard_detect; target_invert_pin_t buzzer; @@ -211,6 +212,7 @@ typedef struct { MEMBER(fpv, gpio_pins_t) \ MEMBER(vbat, gpio_pins_t) \ MEMBER(ibat, gpio_pins_t) \ + MEMBER(rgb_led, gpio_pins_t) \ MEMBER(sdcard_detect, target_invert_pin_t) \ MEMBER(buzzer, target_invert_pin_t) \ ARRAY_MEMBER(motor_pins, MOTOR_PIN_MAX, gpio_pins_t) \ diff --git a/src/driver/at32/dma.c b/src/driver/at32/dma.c index e551d9719..8825f2cad 100644 --- a/src/driver/at32/dma.c +++ b/src/driver/at32/dma.c @@ -1,55 +1,119 @@ #include "driver/dma.h" #include "driver/rcc.h" +#include "driver/resource.h" +#include "driver/timer.h" + +#define DMA_CHANNEL_MAX 14 #define DMA_GL_FLAG ((uint32_t)0x00000001) #define DMA_FDT_FLAG ((uint32_t)0x00000002) #define DMA_HDT_FLAG ((uint32_t)0x00000004) #define DMA_DTERR_FLAG ((uint32_t)0x00000008) -#define DMAMUX_DMAREQ_ID_TIM1_CH1 DMAMUX_DMAREQ_ID_TMR1_CH1 -#define DMAMUX_DMAREQ_ID_TIM1_CH2 DMAMUX_DMAREQ_ID_TMR1_CH2 -#define DMAMUX_DMAREQ_ID_TIM1_CH3 DMAMUX_DMAREQ_ID_TMR1_CH3 -#define DMAMUX_DMAREQ_ID_TIM1_CH4 DMAMUX_DMAREQ_ID_TMR1_CH3 - -#define DMA_STREAMS \ - DMA_STREAM(1, 1, SPI1_RX) \ - DMA_STREAM(1, 2, SPI1_TX) \ - DMA_STREAM(1, 3, SPI2_RX) \ - DMA_STREAM(1, 4, SPI2_TX) \ - DMA_STREAM(1, 5, SPI3_RX) \ - DMA_STREAM(1, 6, SPI3_TX) \ - DMA_STREAM(2, 1, SPI4_RX) \ - DMA_STREAM(2, 2, SPI4_TX) \ - DMA_STREAM(2, 3, TIM1_CH1) \ - DMA_STREAM(2, 4, TIM1_CH3) \ - DMA_STREAM(2, 5, TIM1_CH4) - -#define DMA_STREAM(_port, _chan, _dev) \ - [DMA_DEVICE_##_dev] = { \ - .device = DMA_DEVICE_##_dev, \ - .port = DMA##_port, \ - .port_index = _port, \ - .channel = DMA##_port##_CHANNEL##_chan, \ - .channel_index = _chan, \ - .request = DMAMUX_DMAREQ_ID_##_dev, \ - .mux = DMA##_port##MUX_CHANNEL##_chan, \ - .irq = DMA##_port##_Channel##_chan##_IRQn, \ - }, - -const dma_stream_def_t dma_stream_defs[DMA_DEVICE_MAX] = {DMA_STREAMS}; - -#undef DMA_STREAM - -void dma_prepare_tx_memory(void *addr, uint32_t size) { -} - -void dma_prepare_rx_memory(void *addr, uint32_t size) { -} - -void dma_enable_rcc(dma_device_t dev) { - const dma_stream_def_t *dma = &dma_stream_defs[dev]; - switch (dma->port_index) { +#define DMA_CHANNEL(_port, _chan) \ + { \ + .port = DMA##_port, \ + .port_index = _port, \ + .channel = DMA##_port##_CHANNEL##_chan, \ + .channel_index = _chan, \ + .mux = DMA##_port##MUX_CHANNEL##_chan, \ + .irq = DMA##_port##_Channel##_chan##_IRQn, \ + } + +static const dma_channel_def_t dma_channel_defs[DMA_CHANNEL_MAX] = { + DMA_CHANNEL(1, 1), + DMA_CHANNEL(1, 2), + DMA_CHANNEL(1, 3), + DMA_CHANNEL(1, 4), + DMA_CHANNEL(1, 5), + DMA_CHANNEL(1, 6), + DMA_CHANNEL(1, 7), + DMA_CHANNEL(2, 1), + DMA_CHANNEL(2, 2), + DMA_CHANNEL(2, 3), + DMA_CHANNEL(2, 4), + DMA_CHANNEL(2, 5), + DMA_CHANNEL(2, 6), + DMA_CHANNEL(2, 7), +}; + +#undef DMA_CHANNEL + +typedef struct { + resource_tag_t tag; + dmamux_requst_id_sel_type request; +} dma_req_id_t; + +static const dma_req_id_t dma_req_ids[] = { + {.tag = SPI_TAG(SPI_PORT1, RES_SPI_MISO), .request = DMAMUX_DMAREQ_ID_SPI1_RX}, + {.tag = SPI_TAG(SPI_PORT1, RES_SPI_MOSI), .request = DMAMUX_DMAREQ_ID_SPI1_TX}, + {.tag = SPI_TAG(SPI_PORT2, RES_SPI_MISO), .request = DMAMUX_DMAREQ_ID_SPI2_RX}, + {.tag = SPI_TAG(SPI_PORT2, RES_SPI_MOSI), .request = DMAMUX_DMAREQ_ID_SPI2_TX}, + {.tag = SPI_TAG(SPI_PORT3, RES_SPI_MISO), .request = DMAMUX_DMAREQ_ID_SPI3_RX}, + {.tag = SPI_TAG(SPI_PORT3, RES_SPI_MOSI), .request = DMAMUX_DMAREQ_ID_SPI3_TX}, + {.tag = SPI_TAG(SPI_PORT4, RES_SPI_MISO), .request = DMAMUX_DMAREQ_ID_SPI4_RX}, + {.tag = SPI_TAG(SPI_PORT4, RES_SPI_MOSI), .request = DMAMUX_DMAREQ_ID_SPI4_TX}, + {.tag = TIMER_TAG(TIMER1, TIMER_CH1), .request = DMAMUX_DMAREQ_ID_TMR1_CH1}, + {.tag = TIMER_TAG(TIMER1, TIMER_CH2), .request = DMAMUX_DMAREQ_ID_TMR1_CH2}, + {.tag = TIMER_TAG(TIMER1, TIMER_CH3), .request = DMAMUX_DMAREQ_ID_TMR1_CH3}, + {.tag = TIMER_TAG(TIMER1, TIMER_CH4), .request = DMAMUX_DMAREQ_ID_TMR1_CH4}, + {.tag = TIMER_TAG(TIMER8, TIMER_CH1), .request = DMAMUX_DMAREQ_ID_TMR8_CH1}, + {.tag = TIMER_TAG(TIMER8, TIMER_CH2), .request = DMAMUX_DMAREQ_ID_TMR8_CH2}, + {.tag = TIMER_TAG(TIMER8, TIMER_CH3), .request = DMAMUX_DMAREQ_ID_TMR8_CH3}, + {.tag = TIMER_TAG(TIMER8, TIMER_CH4), .request = DMAMUX_DMAREQ_ID_TMR8_CH4}, + {.tag = TIMER_TAG(TIMER2, TIMER_CH1), .request = DMAMUX_DMAREQ_ID_TMR2_CH1}, + {.tag = TIMER_TAG(TIMER2, TIMER_CH2), .request = DMAMUX_DMAREQ_ID_TMR2_CH2}, + {.tag = TIMER_TAG(TIMER2, TIMER_CH3), .request = DMAMUX_DMAREQ_ID_TMR2_CH3}, + {.tag = TIMER_TAG(TIMER2, TIMER_CH4), .request = DMAMUX_DMAREQ_ID_TMR2_CH4}, + {.tag = TIMER_TAG(TIMER3, TIMER_CH1), .request = DMAMUX_DMAREQ_ID_TMR3_CH1}, + {.tag = TIMER_TAG(TIMER3, TIMER_CH2), .request = DMAMUX_DMAREQ_ID_TMR3_CH2}, + {.tag = TIMER_TAG(TIMER3, TIMER_CH3), .request = DMAMUX_DMAREQ_ID_TMR3_CH3}, + {.tag = TIMER_TAG(TIMER3, TIMER_CH4), .request = DMAMUX_DMAREQ_ID_TMR3_CH4}, + {.tag = TIMER_TAG(TIMER4, TIMER_CH1), .request = DMAMUX_DMAREQ_ID_TMR4_CH1}, + {.tag = TIMER_TAG(TIMER4, TIMER_CH2), .request = DMAMUX_DMAREQ_ID_TMR4_CH2}, + {.tag = TIMER_TAG(TIMER4, TIMER_CH3), .request = DMAMUX_DMAREQ_ID_TMR4_CH3}, + {.tag = TIMER_TAG(TIMER4, TIMER_CH4), .request = DMAMUX_DMAREQ_ID_TMR4_CH4}, + {.tag = TIMER_TAG(TIMER5, TIMER_CH1), .request = DMAMUX_DMAREQ_ID_TMR5_CH1}, + {.tag = TIMER_TAG(TIMER5, TIMER_CH2), .request = DMAMUX_DMAREQ_ID_TMR5_CH2}, + {.tag = TIMER_TAG(TIMER5, TIMER_CH3), .request = DMAMUX_DMAREQ_ID_TMR5_CH3}, + {.tag = TIMER_TAG(TIMER5, TIMER_CH4), .request = DMAMUX_DMAREQ_ID_TMR5_CH4}, + {.tag = TIMER_TAG(TIMER20, TIMER_CH1), .request = DMAMUX_DMAREQ_ID_TMR20_CH1}, + {.tag = TIMER_TAG(TIMER20, TIMER_CH2), .request = DMAMUX_DMAREQ_ID_TMR20_CH2}, + {.tag = TIMER_TAG(TIMER20, TIMER_CH3), .request = DMAMUX_DMAREQ_ID_TMR20_CH3}, + {.tag = TIMER_TAG(TIMER20, TIMER_CH4), .request = DMAMUX_DMAREQ_ID_TMR20_CH4}, +}; + +#define DMA_REQ_ID_MAX (sizeof(dma_req_ids) / sizeof(dma_req_id_t)) + +static dma_assigment_t dma_assigments[DMA_CHANNEL_MAX] = {}; + +static dmamux_requst_id_sel_type dma_find_req(resource_tag_t tag) { + for (uint32_t i = 0; i < DMA_REQ_ID_MAX; i++) { + const dma_req_id_t *req = &dma_req_ids[i]; + if (req->tag == tag) { + return req->request; + } + } + return 0; +} + +const dma_assigment_t *dma_alloc(resource_tag_t tag, dma_device_t dev) { + for (uint32_t i = 0; i < DMA_CHANNEL_MAX; i++) { + dma_assigment_t *ass = &dma_assigments[i]; + const dma_channel_def_t *def = &dma_channel_defs[i]; + if (ass->dev == 0) { + ass->def = def; + ass->request = dma_find_req(tag); + ass->dev = dev; + return ass; + } + } + return NULL; +} + +void dma_enable_rcc(const dma_assigment_t *ass) { + switch (ass->def->port_index) { case 1: rcc_enable(RCC_ENCODE(DMA1)); dmamux_enable(DMA1, TRUE); @@ -61,58 +125,103 @@ void dma_enable_rcc(dma_device_t dev) { } } -static uint32_t dma_flag_for_channel(const dma_stream_def_t *dma, uint32_t val) { +static uint32_t dma_flag_for_channel(const dma_channel_def_t *dma, uint32_t val) { // 4bits per channel const uint32_t shift = (dma->channel_index - 1) * 4; const uint32_t port = dma->port_index == 2 ? 0x10000000 : 0x0; return port | (val << shift); } -bool dma_is_flag_active_tc(dma_device_t dev) { - const dma_stream_def_t *dma = &dma_stream_defs[dev]; - const uint32_t flag = dma_flag_for_channel(dma, DMA_FDT_FLAG); +bool dma_is_flag_active_tc(const dma_assigment_t *ass) { + const uint32_t flag = dma_flag_for_channel(ass->def, DMA_FDT_FLAG); return dma_flag_get(flag); } -void dma_clear_flag_tc(dma_device_t dev) { - const dma_stream_def_t *dma = &dma_stream_defs[dev]; - dma_flag_clear(dma_flag_for_channel(dma, DMA_FDT_FLAG)); - dma_flag_clear(dma_flag_for_channel(dma, DMA_HDT_FLAG)); - dma_flag_clear(dma_flag_for_channel(dma, DMA_DTERR_FLAG)); +void dma_clear_flag_tc(const dma_assigment_t *ass) { + dma_flag_clear(dma_flag_for_channel(ass->def, DMA_FDT_FLAG)); + dma_flag_clear(dma_flag_for_channel(ass->def, DMA_HDT_FLAG)); + dma_flag_clear(dma_flag_for_channel(ass->def, DMA_DTERR_FLAG)); } -extern void dshot_dma_isr(dma_device_t dev); -extern void spi_dma_isr(dma_device_t dev); +void dma_prepare_tx_memory(void *addr, uint32_t size) {} +void dma_prepare_rx_memory(void *addr, uint32_t size) {} + +extern void dshot_dma_isr(const dma_assigment_t *); +extern void spi_dma_isr(const dma_assigment_t *); +extern void rgb_dma_isr(const dma_assigment_t *); + +static void handle_dma_stream_isr(const dma_assigment_t *ass) { + switch (ass->dev) { + case DMA_DEVICE_TIM1_CH1: + case DMA_DEVICE_TIM1_CH3: + case DMA_DEVICE_TIM1_CH4: +#ifdef USE_MOTOR_DSHOT + dshot_dma_isr(ass); +#endif + break; -static void handle_dma_stream_isr(dma_device_t dev) { - switch (dev) { case DMA_DEVICE_SPI1_RX: - case DMA_DEVICE_SPI2_RX: - case DMA_DEVICE_SPI3_RX: - case DMA_DEVICE_SPI4_RX: case DMA_DEVICE_SPI1_TX: + case DMA_DEVICE_SPI2_RX: case DMA_DEVICE_SPI2_TX: + case DMA_DEVICE_SPI3_RX: case DMA_DEVICE_SPI3_TX: + case DMA_DEVICE_SPI4_RX: case DMA_DEVICE_SPI4_TX: - spi_dma_isr(dev); + spi_dma_isr(ass); break; - case DMA_DEVICE_TIM1_CH1: - case DMA_DEVICE_TIM1_CH3: - case DMA_DEVICE_TIM1_CH4: -#ifdef USE_MOTOR_DSHOT - dshot_dma_isr(dev); + + case DMA_DEVICE_RGB: +#ifdef USE_RGB_LED + rgb_dma_isr(ass); #endif break; - case DMA_DEVICE_MAX: + + default: break; } } -#define DMA_STREAM(_port, _chan, _dev) \ - void DMA##_port##_Channel##_chan##_IRQHandler() { \ - handle_dma_stream_isr(DMA_DEVICE_##_dev); \ - } - -DMA_STREAMS +void DMA1_Channel1_IRQHandler() { + handle_dma_stream_isr(&dma_assigments[0]); +} +void DMA1_Channel2_IRQHandler() { + handle_dma_stream_isr(&dma_assigments[1]); +} +void DMA1_Channel3_IRQHandler() { + handle_dma_stream_isr(&dma_assigments[2]); +} +void DMA1_Channel4_IRQHandler() { + handle_dma_stream_isr(&dma_assigments[3]); +} +void DMA1_Channel5_IRQHandler() { + handle_dma_stream_isr(&dma_assigments[4]); +} +void DMA1_Channel6_IRQHandler() { + handle_dma_stream_isr(&dma_assigments[5]); +} +void DMA1_Channel7_IRQHandler() { + handle_dma_stream_isr(&dma_assigments[6]); +} -#undef DMA_STREAM \ No newline at end of file +void DMA2_Channel1_IRQHandler() { + handle_dma_stream_isr(&dma_assigments[7]); +} +void DMA2_Channel2_IRQHandler() { + handle_dma_stream_isr(&dma_assigments[8]); +} +void DMA2_Channel3_IRQHandler() { + handle_dma_stream_isr(&dma_assigments[9]); +} +void DMA2_Channel4_IRQHandler() { + handle_dma_stream_isr(&dma_assigments[10]); +} +void DMA2_Channel5_IRQHandler() { + handle_dma_stream_isr(&dma_assigments[11]); +} +void DMA2_Channel6_IRQHandler() { + handle_dma_stream_isr(&dma_assigments[12]); +} +void DMA2_Channel7_IRQHandler() { + handle_dma_stream_isr(&dma_assigments[13]); +} \ No newline at end of file diff --git a/src/driver/at32/dma.h b/src/driver/at32/dma.h new file mode 100644 index 000000000..2bc5e404c --- /dev/null +++ b/src/driver/at32/dma.h @@ -0,0 +1,42 @@ +#pragma once + +#include "hardware.h" + +#include "driver/resource.h" + +typedef enum { + DMA_DEVICE_INVALID, + + DMA_DEVICE_SPI1_RX, + DMA_DEVICE_SPI1_TX, + DMA_DEVICE_SPI2_RX, + DMA_DEVICE_SPI2_TX, + DMA_DEVICE_SPI3_RX, + DMA_DEVICE_SPI3_TX, + DMA_DEVICE_SPI4_RX, + DMA_DEVICE_SPI4_TX, + DMA_DEVICE_TIM1_CH1, + DMA_DEVICE_TIM1_CH3, + DMA_DEVICE_TIM1_CH4, + DMA_DEVICE_RGB, + + DMA_DEVICE_MAX, +} dma_device_t; + +typedef struct { + dma_type *port; + uint8_t port_index; + + dma_channel_type *channel; + uint8_t channel_index; + + dmamux_channel_type *mux; + + IRQn_Type irq; +} dma_channel_def_t; + +typedef struct { + dma_device_t dev; + const dma_channel_def_t *def; + dmamux_requst_id_sel_type request; +} dma_assigment_t; \ No newline at end of file diff --git a/src/driver/at32/hardware.h b/src/driver/at32/hardware.h new file mode 100644 index 000000000..ad00e5fae --- /dev/null +++ b/src/driver/at32/hardware.h @@ -0,0 +1,16 @@ +#pragma once + +#ifdef AT32F4 +#include +#include +#endif + +#ifdef AT32F435 +#define SYS_CLOCK_FREQ_HZ 288000000 +#define PWM_CLOCK_FREQ_HZ 288000000 +#define SPI_CLOCK_FREQ_HZ (SYS_CLOCK_FREQ_HZ / 2) + +#define LOOPTIME LOOPTIME_8K + +#define UID_BASE 0x1FFFF7E8 +#endif \ No newline at end of file diff --git a/src/driver/at32/motor_dshot.c b/src/driver/at32/motor_dshot.c index a39225f68..03909ead3 100644 --- a/src/driver/at32/motor_dshot.c +++ b/src/driver/at32/motor_dshot.c @@ -12,6 +12,8 @@ #ifdef USE_MOTOR_DSHOT +#define DSHOT_TIMER TIMER1 + #define DSHOT_TIME profile.motor.dshot_time #define DSHOT_SYMBOL_TIME (PWM_CLOCK_FREQ_HZ / (3 * DSHOT_TIME * 1000) - 1) @@ -31,8 +33,9 @@ typedef struct { uint32_t port_low; // motor pins for BSRRL, for setting pins low uint32_t port_high; // motor pins for BSRRH, for setting pins high - uint32_t timer_channel; + timer_channel_t timer_channel; dma_device_t dma_device; + const dma_assigment_t *dma; } dshot_gpio_port_t; extern uint16_t dshot_packet[MOTOR_PIN_MAX]; @@ -43,15 +46,15 @@ volatile uint32_t dshot_dma_phase = 0; // 0: idle, 1 - (gpio_port_count + 1): ha static uint8_t gpio_port_count = 0; static dshot_gpio_port_t gpio_ports[DSHOT_MAX_PORT_COUNT] = { { - .timer_channel = TMR_SELECT_CHANNEL_1, + .timer_channel = TIMER_CH1, .dma_device = DMA_DEVICE_TIM1_CH1, }, { - .timer_channel = TMR_SELECT_CHANNEL_3, + .timer_channel = TIMER_CH3, .dma_device = DMA_DEVICE_TIM1_CH3, }, { - .timer_channel = TMR_SELECT_CHANNEL_4, + .timer_channel = TIMER_CH4, .dma_device = DMA_DEVICE_TIM1_CH4, }, }; @@ -92,7 +95,9 @@ static void dshot_init_motor_pin(uint32_t index) { } static void dshot_init_gpio_port(dshot_gpio_port_t *port) { - dma_enable_rcc(port->dma_device); + port->dma = dma_alloc(TIMER_TAG(DSHOT_TIMER, port->timer_channel), port->dma_device); + + dma_enable_rcc(port->dma); tmr_output_config_type tim_oc_init; tmr_output_default_para_init(&tim_oc_init); @@ -103,10 +108,10 @@ static void dshot_init_gpio_port(dshot_gpio_port_t *port) { tmr_output_channel_config(TMR1, port->timer_channel, &tim_oc_init); tmr_output_channel_buffer_enable(TMR1, port->timer_channel, TRUE); - const dma_stream_def_t *dma = &dma_stream_defs[port->dma_device]; + const dma_channel_def_t *dma = port->dma->def; dma_reset(dma->channel); - dmamux_init(dma->mux, dma->request); + dmamux_init(dma->mux, port->dma->request); dma_init_type init; init.peripheral_base_addr = (uint32_t)(&port->gpio->scr); @@ -125,22 +130,6 @@ static void dshot_init_gpio_port(dshot_gpio_port_t *port) { interrupt_enable(dma->irq, DMA_PRIORITY); } -static void dshot_enable_dma_request(uint32_t timer_channel, confirm_state new_state) { - switch (timer_channel) { - case TMR_SELECT_CHANNEL_1: - tmr_dma_request_enable(TMR1, TMR_C1_DMA_REQUEST, new_state); - break; - case TMR_SELECT_CHANNEL_3: - tmr_dma_request_enable(TMR1, TMR_C3_DMA_REQUEST, new_state); - break; - case TMR_SELECT_CHANNEL_4: - tmr_dma_request_enable(TMR1, TMR_C4_DMA_REQUEST, new_state); - break; - default: - break; - } -} - void motor_dshot_init() { gpio_port_count = 0; @@ -172,16 +161,16 @@ void motor_dshot_init() { static void dshot_dma_setup_port(uint32_t index) { const dshot_gpio_port_t *port = &gpio_ports[index]; - const dma_stream_def_t *dma = &dma_stream_defs[port->dma_device]; + const dma_channel_def_t *dma = port->dma->def; - dma_clear_flag_tc(port->dma_device); + dma_clear_flag_tc(port->dma); dma->channel->paddr = (uint32_t)(&port->gpio->scr); dma->channel->maddr = (uint32_t)&port_dma_buffer[index][0]; dma->channel->dtcnt = DSHOT_DMA_BUFFER_SIZE; dma_channel_enable(dma->channel, TRUE); - dshot_enable_dma_request(port->timer_channel, TRUE); + timer_enable_dma_request(DSHOT_TIMER, port->timer_channel, true); } // make dshot dma packet, then fire @@ -238,18 +227,17 @@ void motor_dshot_wait_for_ready() { __NOP(); } -void dshot_dma_isr(dma_device_t dev) { +void dshot_dma_isr(const dma_assigment_t *ass) { for (uint32_t j = 0; j < gpio_port_count; j++) { const dshot_gpio_port_t *port = &gpio_ports[j]; - if (port->dma_device != dev) { + if (port->dma_device != ass->dev) { continue; } - dma_clear_flag_tc(port->dma_device); + dma_clear_flag_tc(ass); - const dma_stream_def_t *dma = &dma_stream_defs[dev]; - dma_channel_enable(dma->channel, FALSE); - dshot_enable_dma_request(port->timer_channel, FALSE); + dma_channel_enable(ass->def->channel, FALSE); + timer_enable_dma_request(DSHOT_TIMER, port->timer_channel, false); dshot_dma_phase--; break; diff --git a/src/driver/at32/motor_pwm.c b/src/driver/at32/motor_pwm.c index 2f0c45f34..a0d653172 100644 --- a/src/driver/at32/motor_pwm.c +++ b/src/driver/at32/motor_pwm.c @@ -54,6 +54,7 @@ void motor_pwm_init() { tmr_output_channel_config(def->instance, timer_channel_val(ch), &tim_oc_init); tmr_channel_value_set(def->instance, timer_channel_val(ch), 0); tmr_output_channel_buffer_enable(def->instance, timer_channel_val(ch), TRUE); + tmr_channel_enable(def->instance, timer_channel_val(ch), TRUE); tmr_output_enable(def->instance, TRUE); tmr_counter_enable(def->instance, TRUE); diff --git a/src/driver/at32/rgb_led.c b/src/driver/at32/rgb_led.c new file mode 100644 index 000000000..859ddf0ff --- /dev/null +++ b/src/driver/at32/rgb_led.c @@ -0,0 +1,203 @@ +#include "driver/rgb_led.h" + +#include +#include + +#include "core/project.h" +#include "driver/dma.h" +#include "driver/gpio.h" +#include "driver/interrupt.h" +#include "driver/timer.h" + +#if defined(USE_RGB_LED) + +#define DMA_DEV DMA_DEVICE_TIM1_CH2 + +#define TIMER_HZ 48000000 +#define TIMER_DIV ((PWM_CLOCK_FREQ_HZ / TIMER_HZ) - 1) + +#define RGB_BIT_TIME ((TIMER_HZ / 800000) - 1) +#define RGB_T0H_TIME ((RGB_BIT_TIME / 3) + 1) +#define RGB_T1H_TIME ((RGB_BIT_TIME / 3) * 2 + 1) + +#define RGB_BITS_LED 24 +#define RGB_BUFFER_SIZE (RGB_BITS_LED * RGB_LED_MAX + 40) + +static resource_tag_t timer_tag; +static const dma_assigment_t *rgb_dma; + +static volatile bool rgb_dma_busy = false; +static DMA_RAM uint32_t rgb_timer_buffer[RGB_BUFFER_SIZE]; +static uint32_t rgb_timer_buffer_count = 0; + +static uint32_t timer_channel_addr(timer_dev_t *timer, timer_channel_t chan) { + switch (chan) { + case TIMER_CH1: + case TIMER_CH1N: + return (uint32_t)(&timer->c1dt); + case TIMER_CH2: + case TIMER_CH2N: + return (uint32_t)(&timer->c2dt); + case TIMER_CH3: + case TIMER_CH3N: + return (uint32_t)(&timer->c3dt); + case TIMER_CH4: + return (uint32_t)(&timer->c4dt); + default: + return 0; + } +} + +static const gpio_af_t *rgb_led_find_af(gpio_pins_t pin) { + for (uint32_t j = 0; j < GPIO_AF_MAX; j++) { + const gpio_af_t *func = &gpio_pin_afs[j]; + if (func->pin != pin || RESOURCE_TAG_TYPE(func->tag) != RESOURCE_TIM) { + continue; + } + + if (timer_alloc_tag(TIMER_USE_RGB_LED, func->tag)) { + return func; + } + } + + return NULL; +} + +void rgb_led_init() { + const gpio_pins_t pin = target.rgb_led; + if (pin == PIN_NONE) { + return; + } + + const gpio_af_t *func = rgb_led_find_af(pin); + if (func == NULL) { + return; + } + + timer_tag = func->tag; + rgb_dma = dma_alloc(timer_tag, DMA_DEVICE_RGB); + + gpio_config_t gpio_init; + gpio_init.mode = GPIO_ALTERNATE; + gpio_init.drive = GPIO_DRIVE_HIGH; + gpio_init.output = GPIO_PUSHPULL; + gpio_init.pull = GPIO_NO_PULL; + gpio_pin_init_af(pin, gpio_init, func->af); + + const timer_channel_t ch = TIMER_TAG_CH(func->tag); + const timer_index_t tim = TIMER_TAG_TIM(func->tag); + const timer_def_t *def = &timer_defs[tim]; + + rcc_enable(def->rcc); + + tmr_counter_enable(def->instance, FALSE); + + tmr_base_init(def->instance, RGB_BIT_TIME, TIMER_DIV); + tmr_cnt_dir_set(def->instance, TMR_COUNT_UP); + tmr_clock_source_div_set(def->instance, TMR_CLOCK_DIV1); + + tmr_output_config_type tim_oc_init; + tmr_output_default_para_init(&tim_oc_init); + tim_oc_init.oc_mode = TMR_OUTPUT_CONTROL_PWM_MODE_A; + tim_oc_init.oc_idle_state = FALSE; + tim_oc_init.oc_polarity = TMR_OUTPUT_ACTIVE_HIGH; + tim_oc_init.oc_output_state = TRUE; + tmr_output_channel_config(def->instance, timer_channel_val(ch), &tim_oc_init); + + tmr_channel_value_set(def->instance, timer_channel_val(ch), 0); + tmr_output_channel_buffer_enable(def->instance, timer_channel_val(ch), TRUE); + tmr_channel_enable(def->instance, timer_channel_val(ch), TRUE); + + tmr_period_buffer_enable(def->instance, TRUE); + + dma_enable_rcc(rgb_dma); + dma_reset(rgb_dma->def->channel); + + dma_init_type init; + init.peripheral_base_addr = timer_channel_addr(def->instance, ch); + init.memory_base_addr = (uint32_t)rgb_timer_buffer; + init.direction = DMA_DIR_MEMORY_TO_PERIPHERAL; + init.buffer_size = RGB_BUFFER_SIZE; + init.peripheral_inc_enable = FALSE; + init.memory_inc_enable = TRUE; + init.peripheral_data_width = DMA_PERIPHERAL_DATA_WIDTH_WORD; + init.memory_data_width = DMA_MEMORY_DATA_WIDTH_WORD; + init.loop_mode_enable = FALSE; + init.priority = DMA_PRIORITY_MEDIUM; + dma_init(rgb_dma->def->channel, &init); + dmamux_init(rgb_dma->def->mux, rgb_dma->request); + + interrupt_enable(rgb_dma->def->irq, DMA_PRIORITY); + + tmr_output_enable(def->instance, TRUE); +} + +bool rgb_led_busy() { + return rgb_dma_busy; +} + +void rgb_led_set_value(uint32_t value, uint32_t count) { + if (rgb_dma_busy) { + return; + } + + uint32_t offset = 0; + + for (uint32_t i = 0; i < 20; i++) { + rgb_timer_buffer[offset++] = 0; + } + for (uint32_t i = 0; i < count; i++) { + // rgb_led_value contains a (32bit) int that contains the RGB values in G R B format already + // Test each bit and assign the T1H or T0H depending on whether it is 1 or 0. + for (int32_t j = RGB_BITS_LED - 1; j >= 0; j--) { + rgb_timer_buffer[offset++] = ((value >> j) & 0x1) ? RGB_T1H_TIME : RGB_T0H_TIME; + } + } + for (uint32_t i = 0; i < 20; i++) { + rgb_timer_buffer[offset++] = 0; + } + + rgb_timer_buffer_count = offset; +} + +void rgb_led_send() { + if (rgb_dma_busy) { + return; + } + rgb_dma_busy = true; + + dma_prepare_tx_memory((void *)rgb_timer_buffer, sizeof(rgb_timer_buffer)); + dma_clear_flag_tc(rgb_dma); + + const timer_channel_t ch = TIMER_TAG_CH(timer_tag); + const timer_index_t tim = TIMER_TAG_TIM(timer_tag); + const timer_def_t *def = &timer_defs[tim]; + + rgb_dma->def->channel->paddr = timer_channel_addr(def->instance, ch); + rgb_dma->def->channel->maddr = (uint32_t)rgb_timer_buffer; + rgb_dma->def->channel->dtcnt = rgb_timer_buffer_count; + + dma_interrupt_enable(rgb_dma->def->channel, DMA_FDT_INT, TRUE); + + timer_enable_dma_request(tim, ch, TRUE); + tmr_counter_value_set(def->instance, 0); + + tmr_counter_enable(def->instance, TRUE); + dma_channel_enable(rgb_dma->def->channel, TRUE); +} + +void rgb_dma_isr(const dma_assigment_t *ass) { + dma_clear_flag_tc(ass); + dma_channel_enable(ass->def->channel, FALSE); + + const uint8_t ch = TIMER_TAG_CH(timer_tag); + const uint8_t tim = TIMER_TAG_TIM(timer_tag); + const timer_def_t *def = &timer_defs[tim]; + + tmr_counter_enable(def->instance, FALSE); + timer_enable_dma_request(tim, ch, FALSE); + + rgb_dma_busy = false; +} + +#endif \ No newline at end of file diff --git a/src/driver/at32/spi.c b/src/driver/at32/spi.c index a3b257c1f..b21d6da06 100644 --- a/src/driver/at32/spi.c +++ b/src/driver/at32/spi.c @@ -17,29 +17,29 @@ const spi_port_def_t spi_port_defs[SPI_PORT_MAX] = { .channel_index = 1, .channel = SPI1, .rcc = RCC_ENCODE(SPI1), - .dma_rx = DMA_DEVICE_SPI1_RX, - .dma_tx = DMA_DEVICE_SPI1_TX, + .dma_dev_rx = DMA_DEVICE_SPI1_RX, + .dma_dev_tx = DMA_DEVICE_SPI1_TX, }, { .channel_index = 2, .channel = SPI2, .rcc = RCC_ENCODE(SPI2), - .dma_rx = DMA_DEVICE_SPI2_RX, - .dma_tx = DMA_DEVICE_SPI2_TX, + .dma_dev_rx = DMA_DEVICE_SPI2_RX, + .dma_dev_tx = DMA_DEVICE_SPI2_TX, }, { .channel_index = 3, .channel = SPI3, .rcc = RCC_ENCODE(SPI3), - .dma_rx = DMA_DEVICE_SPI3_RX, - .dma_tx = DMA_DEVICE_SPI3_TX, + .dma_dev_rx = DMA_DEVICE_SPI3_RX, + .dma_dev_tx = DMA_DEVICE_SPI3_TX, }, { .channel_index = 4, .channel = SPI4, .rcc = RCC_ENCODE(SPI4), - .dma_rx = DMA_DEVICE_SPI4_RX, - .dma_tx = DMA_DEVICE_SPI4_TX, + .dma_dev_rx = DMA_DEVICE_SPI4_RX, + .dma_dev_tx = DMA_DEVICE_SPI4_TX, }, }; @@ -48,6 +48,7 @@ extern FAST_RAM volatile uint8_t dma_transfer_done[16]; extern FAST_RAM spi_txn_t txn_pool[SPI_TXN_MAX]; #define PORT spi_port_defs[port] +#define CONF spi_port_config[port] static uint32_t spi_divider_to_ll(uint32_t divider) { switch (divider) { @@ -119,10 +120,10 @@ static void spi_init_pins(spi_ports_t port, gpio_pins_t nss) { } static void spi_dma_init_rx(spi_ports_t port) { - const dma_stream_def_t *dma = &dma_stream_defs[PORT.dma_rx]; + const dma_channel_def_t *dma = CONF.dma_rx->def; dma_reset(dma->channel); - dmamux_init(dma->mux, dma->request); + dmamux_init(dma->mux, CONF.dma_rx->request); dma_init_type init; init.peripheral_base_addr = (uint32_t)(&PORT.channel->dt); @@ -141,16 +142,16 @@ static void spi_dma_init_rx(spi_ports_t port) { static void spi_dma_reset_rx(spi_ports_t port, uint8_t *rx_data, uint32_t rx_size) { dma_prepare_rx_memory(rx_data, rx_size); - const dma_stream_def_t *dma = &dma_stream_defs[PORT.dma_rx]; + const dma_channel_def_t *dma = CONF.dma_rx->def; dma->channel->maddr = (uint32_t)rx_data; dma_data_number_set(dma->channel, rx_size); } static void spi_dma_init_tx(spi_ports_t port) { - const dma_stream_def_t *dma = &dma_stream_defs[PORT.dma_tx]; + const dma_channel_def_t *dma = CONF.dma_tx->def; dma_reset(dma->channel); - dmamux_init(dma->mux, dma->request); + dmamux_init(dma->mux, CONF.dma_tx->request); dma_init_type init; init.peripheral_base_addr = (uint32_t)(&PORT.channel->dt); @@ -169,7 +170,7 @@ static void spi_dma_init_tx(spi_ports_t port) { static void spi_dma_reset_tx(spi_ports_t port, uint8_t *tx_data, uint32_t tx_size) { dma_prepare_tx_memory(tx_data, tx_size); - const dma_stream_def_t *dma = &dma_stream_defs[PORT.dma_tx]; + const dma_channel_def_t *dma = CONF.dma_tx->def; dma->channel->maddr = (uint32_t)tx_data; dma_data_number_set(dma->channel, tx_size); } @@ -226,11 +227,11 @@ void spi_reconfigure(spi_bus_device_t *bus) { void spi_dma_transfer_begin(spi_ports_t port, uint8_t *buffer, uint32_t length) { dma_transfer_done[port] = 0; - const dma_stream_def_t *dma_tx = &dma_stream_defs[PORT.dma_tx]; - const dma_stream_def_t *dma_rx = &dma_stream_defs[PORT.dma_rx]; + const dma_channel_def_t *dma_tx = CONF.dma_tx->def; + const dma_channel_def_t *dma_rx = CONF.dma_rx->def; - dma_clear_flag_tc(PORT.dma_rx); - dma_clear_flag_tc(PORT.dma_tx); + dma_clear_flag_tc(CONF.dma_rx); + dma_clear_flag_tc(CONF.dma_tx); spi_dma_reset_rx(port, buffer, length); spi_dma_reset_tx(port, buffer, length); @@ -259,8 +260,13 @@ void spi_bus_device_init(spi_bus_device_t *bus) { spi_enable_rcc(bus->port); const spi_port_def_t *port = &spi_port_defs[bus->port]; - dma_enable_rcc(port->dma_rx); - dma_enable_rcc(port->dma_tx); + volatile spi_port_config_t *conf = &spi_port_config[bus->port]; + + conf->dma_rx = dma_alloc(SPI_TAG(bus->port, RES_SPI_MISO), port->dma_dev_rx); + conf->dma_tx = dma_alloc(SPI_TAG(bus->port, RES_SPI_MOSI), port->dma_dev_tx); + + dma_enable_rcc(conf->dma_rx); + dma_enable_rcc(conf->dma_tx); spi_i2s_reset(port->channel); @@ -281,8 +287,7 @@ void spi_bus_device_init(spi_bus_device_t *bus) { spi_dma_init_rx(bus->port); spi_dma_init_tx(bus->port); - const dma_stream_def_t *dma_rx = &dma_stream_defs[port->dma_rx]; - interrupt_enable(dma_rx->irq, DMA_PRIORITY); + interrupt_enable(conf->dma_rx->def->irq, DMA_PRIORITY); dma_transfer_done[bus->port] = 1; } @@ -340,15 +345,15 @@ void spi_seg_submit_wait_ex(spi_bus_device_t *bus, const spi_txn_segment_t *segs } static void handle_dma_rx_isr(spi_ports_t port) { - const dma_stream_def_t *dma_rx = &dma_stream_defs[PORT.dma_rx]; - const dma_stream_def_t *dma_tx = &dma_stream_defs[PORT.dma_tx]; + const dma_channel_def_t *dma_rx = CONF.dma_rx->def; + const dma_channel_def_t *dma_tx = CONF.dma_tx->def; - if (!dma_is_flag_active_tc(PORT.dma_rx)) { + if (!dma_is_flag_active_tc(CONF.dma_rx)) { return; } - dma_clear_flag_tc(PORT.dma_rx); - dma_clear_flag_tc(PORT.dma_tx); + dma_clear_flag_tc(CONF.dma_rx); + dma_clear_flag_tc(CONF.dma_tx); dma_interrupt_enable(dma_rx->channel, DMA_FDT_INT, FALSE); dma_interrupt_enable(dma_rx->channel, DMA_DTERR_INT, FALSE); @@ -377,8 +382,8 @@ static void handle_dma_rx_isr(spi_ports_t port) { } } -void spi_dma_isr(dma_device_t dev) { - switch (dev) { +void spi_dma_isr(const dma_assigment_t *ass) { + switch (ass->dev) { case DMA_DEVICE_SPI1_RX: handle_dma_rx_isr(SPI_PORT1); break; diff --git a/src/driver/at32/system.h b/src/driver/at32/system.h index 0c51d4373..d6086004c 100644 --- a/src/driver/at32/system.h +++ b/src/driver/at32/system.h @@ -1,39 +1,12 @@ #pragma once -#ifdef AT32F4 -#include -#include -#endif - -#ifdef AT32F435 -#define SYS_CLOCK_FREQ_HZ 288000000 -#define PWM_CLOCK_FREQ_HZ 288000000 -#define SPI_CLOCK_FREQ_HZ (SYS_CLOCK_FREQ_HZ / 2) - -#define LOOPTIME LOOPTIME_8K - -#define UID_BASE 0x1FFFF7E8 -#endif +#include "hardware.h" #include "adc.h" +#include "dma.h" #include "rcc.h" typedef gpio_type gpio_port_t; typedef spi_type spi_port_t; typedef tmr_type timer_dev_t; typedef usart_type usart_dev_t; - -typedef struct { - uint32_t device; - - dma_type *port; - uint8_t port_index; - - dma_channel_type *channel; - uint8_t channel_index; - - uint32_t request; - dmamux_channel_type *mux; - - IRQn_Type irq; -} dma_stream_def_t; \ No newline at end of file diff --git a/src/driver/at32/timer.c b/src/driver/at32/timer.c index 14bb24729..3299b15b9 100644 --- a/src/driver/at32/timer.c +++ b/src/driver/at32/timer.c @@ -108,6 +108,30 @@ uint32_t timer_channel_val(timer_channel_t chan) { } } +void timer_enable_dma_request(timer_index_t tim, timer_channel_t chan, bool state) { + const timer_def_t *def = &timer_defs[tim]; + + switch (chan) { + case TIMER_CH1: + case TIMER_CH1N: + tmr_dma_request_enable(def->instance, TMR_C1_DMA_REQUEST, state ? TRUE : FALSE); + break; + case TIMER_CH2: + case TIMER_CH2N: + tmr_dma_request_enable(def->instance, TMR_C2_DMA_REQUEST, state ? TRUE : FALSE); + break; + case TIMER_CH3: + case TIMER_CH3N: + tmr_dma_request_enable(def->instance, TMR_C3_DMA_REQUEST, state ? TRUE : FALSE); + break; + case TIMER_CH4: + tmr_dma_request_enable(def->instance, TMR_C4_DMA_REQUEST, state ? TRUE : FALSE); + break; + default: + break; + } +} + static void timer_irq_handler() { extern void soft_serial_timer_irq_handler(); soft_serial_timer_irq_handler(); diff --git a/src/driver/dma.h b/src/driver/dma.h index 4c2fe6bfc..d5f85d055 100644 --- a/src/driver/dma.h +++ b/src/driver/dma.h @@ -6,24 +6,6 @@ #define DMA_ALLOC_BUFFER_SIZE 4096 -typedef enum { - DMA_DEVICE_SPI1_RX, - DMA_DEVICE_SPI1_TX, - DMA_DEVICE_SPI2_RX, - DMA_DEVICE_SPI2_TX, - DMA_DEVICE_SPI3_RX, - DMA_DEVICE_SPI3_TX, - DMA_DEVICE_SPI4_RX, - DMA_DEVICE_SPI4_TX, - DMA_DEVICE_TIM1_CH1, - DMA_DEVICE_TIM1_CH3, - DMA_DEVICE_TIM1_CH4, - - DMA_DEVICE_MAX, -} dma_device_t; - -extern const dma_stream_def_t dma_stream_defs[DMA_DEVICE_MAX]; - void *dma_mem_alloc(uint32_t min_size); void *dma_mem_realloc(void *ptr, uint32_t min_size); void dma_mem_free(void *ptr); @@ -31,7 +13,8 @@ void dma_mem_free(void *ptr); void dma_prepare_tx_memory(void *addr, uint32_t size); void dma_prepare_rx_memory(void *addr, uint32_t size); -void dma_enable_rcc(dma_device_t dev); +void dma_enable_rcc(const dma_assigment_t *ass); +const dma_assigment_t *dma_alloc(resource_tag_t tag, dma_device_t dev); -bool dma_is_flag_active_tc(dma_device_t dev); -void dma_clear_flag_tc(dma_device_t dev); \ No newline at end of file +bool dma_is_flag_active_tc(const dma_assigment_t *ass); +void dma_clear_flag_tc(const dma_assigment_t *ass); \ No newline at end of file diff --git a/src/driver/resource.h b/src/driver/resource.h index b24594600..2321cd7c1 100644 --- a/src/driver/resource.h +++ b/src/driver/resource.h @@ -25,6 +25,8 @@ typedef enum { typedef uint32_t resource_tag_t; +#define RESOURCE_TAG_ANY 0xFFFFFFFF + #define RESOURCE_TAG(typ, val) ((uint32_t)((typ) << 24) | (uint32_t)(val)) #define RESOURCE_TAG_TYPE(tag) (resource_type_t)(((tag) >> 24) & 0xFF) diff --git a/src/driver/rgb_led.c b/src/driver/rgb_led.c deleted file mode 100644 index 2456e1d03..000000000 --- a/src/driver/rgb_led.c +++ /dev/null @@ -1,347 +0,0 @@ -//**********************************************WARNING - THIS FILE NOT YET PORTED FROM F0 SILVERWARE*********************************** - -#include "driver/rgb_led.h" - -#include "core/project.h" -#include "driver/spi.h" -#include "driver/time.h" -#include "util/util.h" - -#if (RGB_LED_NUMBER > 0) -void rgb_send(int data); - -#ifdef RGB_LED_DMA - -#define RGB_BIT_TIME ((SYS_CLOCK_FREQ_HZ / 1000 / 800) - 1) -#define RGB_T0H_TIME (RGB_BIT_TIME * 0.30 + 0.05) -#define RGB_T1H_TIME (RGB_BIT_TIME * 0.60 + 0.05) - -extern int rgb_led_value[]; - -volatile int rgb_dma_phase = 0; // 3:rgb data ready - // 2:rgb dma buffer ready - -// wait to be cascaded after dshot, or fired at next frame if no dshot activity -// 1:rgb dma busy -// 0:idle - -const int offset = RGB_PIN > LL_GPIO_PIN_7; - -volatile uint32_t rgb_data_portA[RGB_LED_NUMBER * 24 / 4] = {0}; // DMA buffer: reset output when bit data=0 at TOH timing -volatile uint16_t rgb_portX[1] = {RGB_PIN}; // sum of all rgb pins at port -volatile uint32_t RGB_DATA16[16]; // 4-bit look-up table for dma buffer making - -void rgb_init() { - if ((RGB_PIN == LL_GPIO_PIN_13 || RGB_PIN == LL_GPIO_PIN_14) && RGB_PORT == GPIOA) { - // programming port used - // wait until 2 seconds from powerup passed - while (time_micros() < 2e6) - ; - } - - gpio_config_t GPIO_InitStructure; - - GPIO_InitStructure.mode = GPIO_OUTPUT; - GPIO_InitStructure.output = GPIO_PUSHPULL; - GPIO_InitStructure.pull = GPIO_NO_PULL; - GPIO_InitStructure.drive =GPIO_DRIVE_HIGH; - - GPIO_InitStructure.Pin = RGB_PIN; - LL_GPIO_Init(RGB_PORT, &GPIO_InitStructure); - -#ifndef USE_DSHOT_DMA_DRIVER - // RGB timer/DMA init - // TIM1_UP DMA_CH5: set all output to HIGH at TIM1 update - // TIM1_CH1 DMA_CH2: reset output if data=0 at T0H timing - // TIM1_CH4 DMA_CH4: reset all output at T1H timing - - TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; - TIM_OCInitTypeDef TIM_OCInitStructure; - - TIM_TimeBaseStructInit(&TIM_TimeBaseStructure); - TIM_OCStructInit(&TIM_OCInitStructure); - // TIM1 Periph clock enable - RCC_APB2PeriphClockCmd(RCC_APB2Periph_TIM1, ENABLE); - - /* Time base configuration */ - TIM_TimeBaseStructure.TIM_Period = RGB_BIT_TIME; - TIM_TimeBaseStructure.TIM_Prescaler = 0; - TIM_TimeBaseStructure.TIM_ClockDivision = 0; - TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; - TIM_TimeBaseInit(TIM1, &TIM_TimeBaseStructure); - TIM_ARRPreloadConfig(TIM1, DISABLE); - - /* Timing Mode configuration: Channel 1 */ - TIM_OCInitStructure.OCMode = LL_TIM_OCMODE_FROZEN; - TIM_OCInitStructure.OCState = LL_TIM_OCSTATE_DISABLE; - TIM_OCInitStructure.CompareValue = RGB_T0H_TIME; - LL_TIM_OC_Init(TIM1, LL_TIM_CHANNEL_CH1, &TIM_OCInitStructure); - LL_TIM_OC_DisablePreload(TIM1, LL_TIM_CHANNEL_CH1); - - /* Timing Mode configuration: Channel 4 */ - TIM_OCInitStructure.OCMode = LL_TIM_OCMODE_FROZEN; - TIM_OCInitStructure.OCState = LL_TIM_OCSTATE_DISABLE; - TIM_OCInitStructure.CompareValue = RGB_T1H_TIME; - LL_TIM_OC_Init(TIM1, LL_TIM_CHANNEL_CH4, &TIM_OCInitStructure); - LL_TIM_OC_DisablePreload(TIM1, LL_TIM_CHANNEL_CH4); - - LL_LL_DMA_InitTypeDef DMA_InitStructure; - - LL_DMA_StructInit(&DMA_InitStructure); - RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); - - /* DMA1 Channe5 configuration ----------------------------------------------*/ - LL_DMA_DeInit(DMA1_Channel3); - DMA_InitStructure.PeriphOrM2MSrcAddress = (uint32_t)&RGB_PORT->BSRR; - DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)rgb_portX; - DMA_InitStructure.Direction = DMA_DIR_PeripheralDST; - DMA_InitStructure.NbData = RGB_LED_NUMBER * 24; - DMA_InitStructure.PeriphOrM2MSrcIncMode = LL_DMA_PERIPH_NOINCREMENT; - DMA_InitStructure.MemoryOrM2MDstIncMode = LL_DMA_MEMORY_NOINCREMENT; - DMA_InitStructure.PeriphOrM2MSrcDataSize = LL_DMA_PDATAALIGN_HALFWORD; - DMA_InitStructure.MemoryOrM2MDstDataSize = LL_DMA_MDATAALIGN_HALFWORD; - DMA_InitStructure.Mode = LL_DMA_MODE_NORMAL; - DMA_InitStructure.Priority = LL_DMA_PRIORITY_HIGH; - DMA_InitStructure.DMA_M2M = DMA_M2M_Disable; - DMA_Init(DMA1_Channel5, &DMA_InitStructure); - - /* DMA1 Channel2 configuration ----------------------------------------------*/ - LL_DMA_DeInit(DMA1_Channel2); - DMA_InitStructure.PeriphOrM2MSrcAddress = (uint32_t)&RGB_PORT->BRR + offset; - DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)rgb_data_portA; - DMA_InitStructure.Direction = DMA_DIR_PeripheralDST; - DMA_InitStructure.NbData = RGB_LED_NUMBER * 24; - DMA_InitStructure.PeriphOrM2MSrcIncMode = LL_DMA_PERIPH_NOINCREMENT; - DMA_InitStructure.MemoryOrM2MDstIncMode = LL_DMA_MEMORY_INCREMENT; - DMA_InitStructure.PeriphOrM2MSrcDataSize = LL_DMA_PDATAALIGN_BYTE; - DMA_InitStructure.MemoryOrM2MDstDataSize = LL_DMA_MDATAALIGN_BYTE; - DMA_InitStructure.Mode = LL_DMA_MODE_NORMAL; - DMA_InitStructure.Priority = LL_DMA_PRIORITY_HIGH; - DMA_InitStructure.DMA_M2M = DMA_M2M_Disable; - DMA_Init(DMA1_Channel2, &DMA_InitStructure); - - /* DMA1 Channel4 configuration ----------------------------------------------*/ - LL_DMA_DeInit(DMA1_Channel4); - DMA_InitStructure.PeriphOrM2MSrcAddress = (uint32_t)&RGB_PORT->BRR; - DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)rgb_portX; - DMA_InitStructure.Direction = DMA_DIR_PeripheralDST; - DMA_InitStructure.NbData = RGB_LED_NUMBER * 24; - DMA_InitStructure.PeriphOrM2MSrcIncMode = LL_DMA_PERIPH_NOINCREMENT; - DMA_InitStructure.MemoryOrM2MDstIncMode = LL_DMA_MEMORY_NOINCREMENT; - DMA_InitStructure.PeriphOrM2MSrcDataSize = LL_DMA_PDATAALIGN_HALFWORD; - DMA_InitStructure.MemoryOrM2MDstDataSize = LL_DMA_MDATAALIGN_HALFWORD; - DMA_InitStructure.Mode = LL_DMA_MODE_NORMAL; - DMA_InitStructure.Priority = LL_DMA_PRIORITY_HIGH; - DMA_InitStructure.DMA_M2M = DMA_M2M_Disable; - DMA_Init(DMA1_Channel4, &DMA_InitStructure); - - TIM_DMACmd(TIM1, TIM_DMA_Update | TIM_DMA_CC4 | TIM_DMA_CC1, ENABLE); - - NVIC_InitTypeDef NVIC_InitStructure; - /* configure DMA1 Channel4 interrupt */ - NVIC_InitStructure.NVIC_IRQChannel = DMA1_Channel4_5_IRQn; - NVIC_InitStructure.NVIC_IRQChannelPriority = (uint8_t)LL_DMA_PRIORITY_HIGH; - NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; - NVIC_Init(&NVIC_InitStructure); - /* enable DMA1 Channel4 transfer complete interrupt */ - DMA_ITConfig(DMA1_Channel4, DMA_IT_TC, ENABLE); -#endif - - for (int i = 0; i < RGB_LED_NUMBER; i++) { - rgb_led_value[i] = 0; - } - if (!rgb_dma_phase) - rgb_dma_phase = 3; - - int pin = rgb_portX[0]; - - if (offset) { - pin >>= 8; - } - - for (int i = 0; i < 16; i++) { - RGB_DATA16[i] = (i & 0x01) ? 0 : pin; - RGB_DATA16[i] <<= 8; - RGB_DATA16[i] |= (i & 0x02) ? 0 : pin; - RGB_DATA16[i] <<= 8; - RGB_DATA16[i] |= (i & 0x04) ? 0 : pin; - RGB_DATA16[i] <<= 8; - RGB_DATA16[i] |= (i & 0x08) ? 0 : pin; - } -} - -void rgb_dma_buffer_making() { - // generate rgb dma packet - int j = 0; - for (int n = 0; n < RGB_LED_NUMBER; n++) { - rgb_data_portA[j++] = RGB_DATA16[(rgb_led_value[n] >> 20) & 0x0f]; - rgb_data_portA[j++] = RGB_DATA16[(rgb_led_value[n] >> 16) & 0x0f]; - rgb_data_portA[j++] = RGB_DATA16[(rgb_led_value[n] >> 12) & 0x0f]; - rgb_data_portA[j++] = RGB_DATA16[(rgb_led_value[n] >> 8) & 0x0f]; - rgb_data_portA[j++] = RGB_DATA16[(rgb_led_value[n] >> 4) & 0x0f]; - rgb_data_portA[j++] = RGB_DATA16[(rgb_led_value[n]) & 0x0f]; - } -} - -void rgb_dma_trigger() { - TIM1->ARR = RGB_BIT_TIME; - TIM1->CCR1 = RGB_T0H_TIME; - TIM1->CCR4 = RGB_T1H_TIME; - - DMA1_Channel5->CPAR = (uint32_t)&RGB_PORT->BSRR; - DMA1_Channel5->CMAR = (uint32_t)rgb_portX; - DMA1_Channel2->CPAR = (uint32_t)&RGB_PORT->BRR + offset; - DMA1_Channel2->CMAR = (uint32_t)rgb_data_portA; - DMA1_Channel4->CPAR = (uint32_t)&RGB_PORT->BRR; - DMA1_Channel4->CMAR = (uint32_t)rgb_portX; - - DMA1_Channel2->CCR &= ~(DMA_CCR_MSIZE_0 | DMA_CCR_MSIZE_1 | DMA_CCR_PSIZE_0 | DMA_CCR_PSIZE_1); // switch from halfword to byte - - DMA_ClearFlag(DMA1_FLAG_GL2 | DMA1_FLAG_GL4 | DMA1_FLAG_GL5); - - DMA1_Channel5->CNDTR = RGB_LED_NUMBER * 24; - DMA1_Channel2->CNDTR = RGB_LED_NUMBER * 24; - DMA1_Channel4->CNDTR = RGB_LED_NUMBER * 24; - - TIM1->SR = 0; - - DMA_Cmd(DMA1_Channel2, ENABLE); - DMA_Cmd(DMA1_Channel4, ENABLE); - DMA_Cmd(DMA1_Channel5, ENABLE); - - TIM_DMACmd(TIM1, TIM_DMA_Update | TIM_DMA_CC4 | TIM_DMA_CC1, ENABLE); - - TIM_SetCounter(TIM1, RGB_BIT_TIME); - TIM_Cmd(TIM1, ENABLE); -} - -void rgb_dma_start() { - if (rgb_dma_phase <= 1) - return; - - if (rgb_dma_phase == 3) { - rgb_dma_buffer_making(); - rgb_dma_phase = 2; - return; - } - -#ifdef USE_DSHOT_DMA_DRIVER - extern int dshot_dma_phase; - if (dshot_dma_phase) - return; -#endif - - rgb_dma_phase = 1; - rgb_dma_trigger(); -} - -void rgb_send(int data) { - if (!rgb_dma_phase) - rgb_dma_phase = 3; -} - -// if dshot dma is used the routine is in that file -#if !defined(USE_DSHOT_DMA_DRIVER) && defined(RGB_LED_DMA) && (RGB_LED_NUMBER > 0) - -void DMA1_Channel4_5_IRQHandler() { - DMA_Cmd(DMA1_Channel5, DISABLE); - DMA_Cmd(DMA1_Channel2, DISABLE); - DMA_Cmd(DMA1_Channel4, DISABLE); - - TIM_DMACmd(TIM1, TIM_DMA_Update | TIM_DMA_CC4 | TIM_DMA_CC1, DISABLE); - DMA_ClearITPendingBit(DMA1_IT_TC4); - TIM_Cmd(TIM1, DISABLE); - - rgb_dma_phase = 0; -} -#endif - -#else - -// sets all leds to a brightness -void rgb_init() { - // spi port inits - - if ((RGB_PIN == LL_GPIO_PIN_13 || RGB_PIN == LL_GPIO_PIN_14) && RGB_PORT == GPIOA) { - // programming port used - - // wait until 2 seconds from powerup passed - while (time_micros() < 2e6) - ; - } - - gpio_config_t GPIO_InitStructure; - - GPIO_InitStructure.mode = GPIO_OUTPUT; - GPIO_InitStructure.output = GPIO_PUSHPULL; - GPIO_InitStructure.pull = GPIO_UP_PULL; - GPIO_InitStructure.drive =GPIO_DRIVE_HIGH; - - GPIO_InitStructure.Pin = RGB_PIN; - LL_GPIO_Init(RGB_PORT, &GPIO_InitStructure); - - for (int i = 0; i < RGB_LED_NUMBER; i++) { - rgb_send(0); - } -} - -#define RGBHIGH gpioset(RGB_PORT, RGB_PIN) -#define RGBLOW gpioreset(RGB_PORT, RGB_PIN); - -#pragma push - -#pragma Otime -#pragma O2 - -void delay1a() { - uint8_t count = 2; - while (count--) - ; -} - -void delay1b() { - uint8_t count = 2; - while (count--) - ; -} - -void delay2a() { - uint8_t count = 1; - while (count--) - ; -} - -void delay2b() { - uint8_t count = 3; - while (count--) - ; -} - -void rgb_send(int data) { - for (int i = 23; i >= 0; i--) { - if ((data >> i) & 1) { - RGBHIGH; - delay1a(); - RGBLOW; - delay1b(); - } else { - RGBHIGH; - delay2a(); - RGBLOW; - delay2b(); - } - } -} -#pragma pop - -#endif - -#else -// rgb led not found -// some dummy headers just in case -void rgb_init() { -} - -void rgb_send(int data) { -} -#endif diff --git a/src/driver/rgb_led.h b/src/driver/rgb_led.h index ded3d5ec5..a52dba850 100644 --- a/src/driver/rgb_led.h +++ b/src/driver/rgb_led.h @@ -1,6 +1,11 @@ #pragma once -void rgb_init(); -void rgb_send(int data); +#include +#include -void rgb_dma_start(); \ No newline at end of file +#define RGB_LED_MAX 32 + +void rgb_led_init(); +void rgb_led_set_value(uint32_t value, uint32_t count); +void rgb_led_send(); +bool rgb_led_busy(); \ No newline at end of file diff --git a/src/driver/spi.h b/src/driver/spi.h index e9edf8862..f40ca3485 100644 --- a/src/driver/spi.h +++ b/src/driver/spi.h @@ -18,8 +18,8 @@ typedef struct { uint8_t channel_index; spi_port_t *channel; rcc_reg_t rcc; - dma_device_t dma_rx; - dma_device_t dma_tx; + dma_device_t dma_dev_rx; + dma_device_t dma_dev_tx; } spi_port_def_t; #define SPI_TXN_MAX 32 @@ -107,6 +107,9 @@ typedef struct { spi_bus_device_t *active_device; spi_mode_t mode; uint32_t hz; + + const dma_assigment_t *dma_rx; + const dma_assigment_t *dma_tx; } spi_port_config_t; extern const spi_port_def_t spi_port_defs[SPI_PORT_MAX]; diff --git a/src/driver/stm32/dma.c b/src/driver/stm32/dma.c index 8dcc3b0a2..3d10289fb 100644 --- a/src/driver/stm32/dma.c +++ b/src/driver/stm32/dma.c @@ -1,75 +1,102 @@ #include "driver/dma.h" #include "driver/rcc.h" +#include "driver/resource.h" +#include "driver/timer.h" -// DMA1 Stream0 SPI3_RX -// DMA1 Stream1 -// DMA1 Stream2 -// DMA1 Stream3 SPI2_RX -// DMA1 Stream4 SPI2_TX -// DMA1 Stream5 -// DMA1 Stream6 -// DMA1 Stream7 SPI3_TX - -// DMA2 Stream0 SPI4_RX -// DMA2 Stream1 SPI4_TX -// DMA2 Stream2 SPI1_RX -// DMA2 Stream3 TIM1_CH1 -// DMA2 Stream4 TIM1_CH4 -// DMA2 Stream5 SPI1_TX -// DMA2 Stream6 TIM1_CH3 -// DMA2 Stream7 - -#define DMA_STREAMS \ - DMA_STREAM(2, 3, 2, SPI1_RX) \ - DMA_STREAM(2, 3, 5, SPI1_TX) \ - DMA_STREAM(1, 0, 3, SPI2_RX) \ - DMA_STREAM(1, 0, 4, SPI2_TX) \ - DMA_STREAM(1, 0, 0, SPI3_RX) \ - DMA_STREAM(1, 0, 7, SPI3_TX) \ - DMA_STREAM(2, 4, 0, SPI4_RX) \ - DMA_STREAM(2, 4, 1, SPI4_TX) \ - DMA_STREAM(2, 6, 3, TIM1_CH1) \ - DMA_STREAM(2, 6, 6, TIM1_CH3) \ - DMA_STREAM(2, 6, 4, TIM1_CH4) - -#ifdef STM32H7 -#define DMA_STREAM(_port, _chan, _stream, _dev) \ - [DMA_DEVICE_##_dev] = { \ - .device = DMA_DEVICE_##_dev, \ - .port = DMA##_port, \ - .port_index = _port, \ - .channel = -1, \ - .channel_index = -1, \ - .request = LL_DMAMUX1_REQ_##_dev, \ - .stream = DMA##_port##_Stream##_stream, \ - .stream_index = LL_DMA_STREAM_##_stream, \ - .irq = DMA##_port##_Stream##_stream##_IRQn, \ - }, -#else -#define DMA_STREAM(_port, _chan, _stream, _dev) \ - [DMA_DEVICE_##_dev] = { \ - .device = DMA_DEVICE_##_dev, \ - .port = DMA##_port, \ - .port_index = _port, \ - .channel = LL_DMA_CHANNEL_##_chan, \ - .channel_index = _chan, \ - .request = -1, \ - .stream = DMA##_port##_Stream##_stream, \ - .stream_index = LL_DMA_STREAM_##_stream, \ - .irq = DMA##_port##_Stream##_stream##_IRQn, \ - }, -#endif - -const dma_stream_def_t dma_stream_defs[DMA_DEVICE_MAX] = {DMA_STREAMS}; - -#undef DMA_STREAM +#define DMA_STREAM_MAX 16 #if defined(STM32F7) || defined(STM32H7) #define CACHE_LINE_SIZE 32 #define CACHE_LINE_MASK (CACHE_LINE_SIZE - 1) #endif +#define DMA_STREAM(_port, _stream) \ + { \ + .port = DMA##_port, \ + .port_index = _port, \ + .stream = DMA##_port##_Stream##_stream, \ + .stream_index = LL_DMA_STREAM_##_stream, \ + .irq = DMA##_port##_Stream##_stream##_IRQn, \ + } +static const dma_stream_def_t dma_stream_defs[DMA_STREAM_MAX] = { + DMA_STREAM(1, 0), + DMA_STREAM(1, 1), + DMA_STREAM(1, 2), + DMA_STREAM(1, 3), + DMA_STREAM(1, 4), + DMA_STREAM(1, 5), + DMA_STREAM(1, 6), + DMA_STREAM(1, 7), + DMA_STREAM(2, 0), + DMA_STREAM(2, 1), + DMA_STREAM(2, 2), + DMA_STREAM(2, 3), + DMA_STREAM(2, 4), + DMA_STREAM(2, 5), + DMA_STREAM(2, 6), + DMA_STREAM(2, 7), +}; +#undef DMA_STREAM + +typedef struct { + resource_tag_t tag; + uint8_t port_index; + uint8_t stream_index; + uint32_t channel; +} dma_channel_t; + +static const dma_channel_t dma_channels[] = { + {.tag = SPI_TAG(SPI_PORT1, RES_SPI_MISO), .port_index = 2, .stream_index = 0, .channel = LL_DMA_CHANNEL_3}, + {.tag = SPI_TAG(SPI_PORT1, RES_SPI_MISO), .port_index = 2, .stream_index = 2, .channel = LL_DMA_CHANNEL_3}, + {.tag = SPI_TAG(SPI_PORT1, RES_SPI_MOSI), .port_index = 2, .stream_index = 3, .channel = LL_DMA_CHANNEL_3}, + {.tag = SPI_TAG(SPI_PORT1, RES_SPI_MOSI), .port_index = 2, .stream_index = 5, .channel = LL_DMA_CHANNEL_3}, + {.tag = SPI_TAG(SPI_PORT2, RES_SPI_MISO), .port_index = 1, .stream_index = 3, .channel = LL_DMA_CHANNEL_0}, + {.tag = SPI_TAG(SPI_PORT2, RES_SPI_MOSI), .port_index = 1, .stream_index = 4, .channel = LL_DMA_CHANNEL_0}, + {.tag = SPI_TAG(SPI_PORT3, RES_SPI_MISO), .port_index = 1, .stream_index = 0, .channel = LL_DMA_CHANNEL_0}, + {.tag = SPI_TAG(SPI_PORT3, RES_SPI_MISO), .port_index = 1, .stream_index = 3, .channel = LL_DMA_CHANNEL_0}, + {.tag = SPI_TAG(SPI_PORT3, RES_SPI_MOSI), .port_index = 1, .stream_index = 5, .channel = LL_DMA_CHANNEL_0}, + {.tag = SPI_TAG(SPI_PORT3, RES_SPI_MOSI), .port_index = 1, .stream_index = 7, .channel = LL_DMA_CHANNEL_0}, + {.tag = SPI_TAG(SPI_PORT4, RES_SPI_MISO), .port_index = 2, .stream_index = 0, .channel = LL_DMA_CHANNEL_4}, + {.tag = SPI_TAG(SPI_PORT4, RES_SPI_MISO), .port_index = 2, .stream_index = 3, .channel = LL_DMA_CHANNEL_5}, + {.tag = SPI_TAG(SPI_PORT4, RES_SPI_MOSI), .port_index = 2, .stream_index = 1, .channel = LL_DMA_CHANNEL_4}, + {.tag = SPI_TAG(SPI_PORT4, RES_SPI_MOSI), .port_index = 2, .stream_index = 4, .channel = LL_DMA_CHANNEL_5}, + + {.tag = TIMER_TAG(TIMER1, TIMER_CH1), .port_index = 2, .stream_index = 4, .channel = LL_DMA_CHANNEL_5}, + {.tag = TIMER_TAG(TIMER1, TIMER_CH2), .port_index = 2, .stream_index = 4, .channel = LL_DMA_CHANNEL_5}, + {.tag = TIMER_TAG(TIMER1, TIMER_CH3), .port_index = 2, .stream_index = 4, .channel = LL_DMA_CHANNEL_5}, + {.tag = TIMER_TAG(TIMER1, TIMER_CH4), .port_index = 2, .stream_index = 4, .channel = LL_DMA_CHANNEL_5}, + + {.tag = TIMER_TAG(TIMER2, TIMER_CH1), .port_index = 2, .stream_index = 4, .channel = LL_DMA_CHANNEL_5}, + {.tag = TIMER_TAG(TIMER2, TIMER_CH2), .port_index = 2, .stream_index = 4, .channel = LL_DMA_CHANNEL_5}, + {.tag = TIMER_TAG(TIMER2, TIMER_CH3), .port_index = 2, .stream_index = 4, .channel = LL_DMA_CHANNEL_5}, + {.tag = TIMER_TAG(TIMER2, TIMER_CH4), .port_index = 2, .stream_index = 4, .channel = LL_DMA_CHANNEL_5}, + + {.tag = TIMER_TAG(TIMER3, TIMER_CH1), .port_index = 2, .stream_index = 4, .channel = LL_DMA_CHANNEL_5}, + {.tag = TIMER_TAG(TIMER3, TIMER_CH2), .port_index = 2, .stream_index = 4, .channel = LL_DMA_CHANNEL_5}, + {.tag = TIMER_TAG(TIMER3, TIMER_CH3), .port_index = 2, .stream_index = 4, .channel = LL_DMA_CHANNEL_5}, + {.tag = TIMER_TAG(TIMER3, TIMER_CH4), .port_index = 2, .stream_index = 4, .channel = LL_DMA_CHANNEL_5}, + + {.tag = TIMER_TAG(TIMER4, TIMER_CH1), .port_index = 2, .stream_index = 4, .channel = LL_DMA_CHANNEL_5}, + {.tag = TIMER_TAG(TIMER4, TIMER_CH2), .port_index = 2, .stream_index = 4, .channel = LL_DMA_CHANNEL_5}, + {.tag = TIMER_TAG(TIMER4, TIMER_CH3), .port_index = 2, .stream_index = 4, .channel = LL_DMA_CHANNEL_5}, + {.tag = TIMER_TAG(TIMER4, TIMER_CH4), .port_index = 2, .stream_index = 4, .channel = LL_DMA_CHANNEL_5}, + + {.tag = TIMER_TAG(TIMER5, TIMER_CH1), .port_index = 2, .stream_index = 4, .channel = LL_DMA_CHANNEL_5}, + {.tag = TIMER_TAG(TIMER5, TIMER_CH2), .port_index = 2, .stream_index = 4, .channel = LL_DMA_CHANNEL_5}, + {.tag = TIMER_TAG(TIMER5, TIMER_CH3), .port_index = 2, .stream_index = 4, .channel = LL_DMA_CHANNEL_5}, + {.tag = TIMER_TAG(TIMER5, TIMER_CH4), .port_index = 2, .stream_index = 4, .channel = LL_DMA_CHANNEL_5}, + + {.tag = TIMER_TAG(TIMER8, TIMER_CH1), .port_index = 2, .stream_index = 4, .channel = LL_DMA_CHANNEL_5}, + {.tag = TIMER_TAG(TIMER8, TIMER_CH2), .port_index = 2, .stream_index = 4, .channel = LL_DMA_CHANNEL_5}, + {.tag = TIMER_TAG(TIMER8, TIMER_CH3), .port_index = 2, .stream_index = 4, .channel = LL_DMA_CHANNEL_5}, + {.tag = TIMER_TAG(TIMER8, TIMER_CH4), .port_index = 2, .stream_index = 4, .channel = LL_DMA_CHANNEL_5}, +}; + +#define DMA_CHANNEL_MAX (sizeof(dma_channels) / sizeof(dma_channel_t)) + +static dma_assigment_t dma_assigments[DMA_STREAM_MAX] = {}; + void dma_prepare_tx_memory(void *addr, uint32_t size) { #if defined(STM32F7) || defined(STM32H7) if (!WITHIN_DTCM_RAM(addr) && !WITHIN_DMA_RAM(addr)) { @@ -86,9 +113,8 @@ void dma_prepare_rx_memory(void *addr, uint32_t size) { #endif } -void dma_enable_rcc(dma_device_t dev) { - const dma_stream_def_t *dma = &dma_stream_defs[dev]; - switch (dma->port_index) { +void dma_enable_rcc(const dma_assigment_t *ass) { + switch (ass->def->port_index) { case 1: rcc_enable(RCC_AHB1_GRP1(DMA1)); break; @@ -98,90 +124,95 @@ void dma_enable_rcc(dma_device_t dev) { } } -bool dma_is_flag_active_tc(dma_device_t dev) { - const dma_stream_def_t *dma = &dma_stream_defs[dev]; - - switch (dma->stream_index) { +bool dma_is_flag_active_tc(const dma_assigment_t *ass) { + switch (ass->def->stream_index) { case LL_DMA_STREAM_0: - return LL_DMA_IsActiveFlag_TC0(dma->port); + return LL_DMA_IsActiveFlag_TC0(ass->def->port); case LL_DMA_STREAM_1: - return LL_DMA_IsActiveFlag_TC1(dma->port); + return LL_DMA_IsActiveFlag_TC1(ass->def->port); case LL_DMA_STREAM_2: - return LL_DMA_IsActiveFlag_TC2(dma->port); + return LL_DMA_IsActiveFlag_TC2(ass->def->port); case LL_DMA_STREAM_3: - return LL_DMA_IsActiveFlag_TC3(dma->port); + return LL_DMA_IsActiveFlag_TC3(ass->def->port); case LL_DMA_STREAM_4: - return LL_DMA_IsActiveFlag_TC4(dma->port); + return LL_DMA_IsActiveFlag_TC4(ass->def->port); case LL_DMA_STREAM_5: - return LL_DMA_IsActiveFlag_TC5(dma->port); + return LL_DMA_IsActiveFlag_TC5(ass->def->port); case LL_DMA_STREAM_6: - return LL_DMA_IsActiveFlag_TC6(dma->port); + return LL_DMA_IsActiveFlag_TC6(ass->def->port); case LL_DMA_STREAM_7: - return LL_DMA_IsActiveFlag_TC7(dma->port); + return LL_DMA_IsActiveFlag_TC7(ass->def->port); } return 0; } -void dma_clear_flag_tc(dma_device_t dev) { - const dma_stream_def_t *dma = &dma_stream_defs[dev]; - - switch (dma->stream_index) { +void dma_clear_flag_tc(const dma_assigment_t *ass) { + switch (ass->def->stream_index) { case LL_DMA_STREAM_0: - LL_DMA_ClearFlag_TC0(dma->port); - LL_DMA_ClearFlag_TE0(dma->port); - LL_DMA_ClearFlag_HT0(dma->port); - LL_DMA_ClearFlag_FE0(dma->port); + LL_DMA_ClearFlag_TC0(ass->def->port); + LL_DMA_ClearFlag_TE0(ass->def->port); + LL_DMA_ClearFlag_HT0(ass->def->port); + LL_DMA_ClearFlag_FE0(ass->def->port); break; case LL_DMA_STREAM_1: - LL_DMA_ClearFlag_TC1(dma->port); - LL_DMA_ClearFlag_TE1(dma->port); - LL_DMA_ClearFlag_HT1(dma->port); - LL_DMA_ClearFlag_FE1(dma->port); + LL_DMA_ClearFlag_TC1(ass->def->port); + LL_DMA_ClearFlag_TE1(ass->def->port); + LL_DMA_ClearFlag_HT1(ass->def->port); + LL_DMA_ClearFlag_FE1(ass->def->port); break; case LL_DMA_STREAM_2: - LL_DMA_ClearFlag_TC2(dma->port); - LL_DMA_ClearFlag_TE2(dma->port); - LL_DMA_ClearFlag_HT2(dma->port); - LL_DMA_ClearFlag_FE2(dma->port); + LL_DMA_ClearFlag_TC2(ass->def->port); + LL_DMA_ClearFlag_TE2(ass->def->port); + LL_DMA_ClearFlag_HT2(ass->def->port); + LL_DMA_ClearFlag_FE2(ass->def->port); break; case LL_DMA_STREAM_3: - LL_DMA_ClearFlag_TC3(dma->port); - LL_DMA_ClearFlag_TE3(dma->port); - LL_DMA_ClearFlag_HT3(dma->port); - LL_DMA_ClearFlag_FE3(dma->port); + LL_DMA_ClearFlag_TC3(ass->def->port); + LL_DMA_ClearFlag_TE3(ass->def->port); + LL_DMA_ClearFlag_HT3(ass->def->port); + LL_DMA_ClearFlag_FE3(ass->def->port); break; case LL_DMA_STREAM_4: - LL_DMA_ClearFlag_TC4(dma->port); - LL_DMA_ClearFlag_TE4(dma->port); - LL_DMA_ClearFlag_HT4(dma->port); - LL_DMA_ClearFlag_FE4(dma->port); + LL_DMA_ClearFlag_TC4(ass->def->port); + LL_DMA_ClearFlag_TE4(ass->def->port); + LL_DMA_ClearFlag_HT4(ass->def->port); + LL_DMA_ClearFlag_FE4(ass->def->port); break; case LL_DMA_STREAM_5: - LL_DMA_ClearFlag_TC5(dma->port); - LL_DMA_ClearFlag_TE5(dma->port); - LL_DMA_ClearFlag_HT5(dma->port); - LL_DMA_ClearFlag_FE5(dma->port); + LL_DMA_ClearFlag_TC5(ass->def->port); + LL_DMA_ClearFlag_TE5(ass->def->port); + LL_DMA_ClearFlag_HT5(ass->def->port); + LL_DMA_ClearFlag_FE5(ass->def->port); break; case LL_DMA_STREAM_6: - LL_DMA_ClearFlag_TC6(dma->port); - LL_DMA_ClearFlag_TE6(dma->port); - LL_DMA_ClearFlag_HT6(dma->port); - LL_DMA_ClearFlag_FE6(dma->port); + LL_DMA_ClearFlag_TC6(ass->def->port); + LL_DMA_ClearFlag_TE6(ass->def->port); + LL_DMA_ClearFlag_HT6(ass->def->port); + LL_DMA_ClearFlag_FE6(ass->def->port); break; case LL_DMA_STREAM_7: - LL_DMA_ClearFlag_TC7(dma->port); - LL_DMA_ClearFlag_TE7(dma->port); - LL_DMA_ClearFlag_HT7(dma->port); - LL_DMA_ClearFlag_FE7(dma->port); + LL_DMA_ClearFlag_TC7(ass->def->port); + LL_DMA_ClearFlag_TE7(ass->def->port); + LL_DMA_ClearFlag_HT7(ass->def->port); + LL_DMA_ClearFlag_FE7(ass->def->port); break; } } -extern void dshot_dma_isr(dma_device_t dev); -extern void spi_dma_isr(dma_device_t dev); +extern void dshot_dma_isr(const dma_assigment_t *); +extern void spi_dma_isr(const dma_assigment_t *); +extern void rgb_dma_isr(const dma_assigment_t *); + +static void handle_dma_stream_isr(const dma_assigment_t *ass) { + switch (ass->dev) { + case DMA_DEVICE_TIM1_CH1: + case DMA_DEVICE_TIM1_CH3: + case DMA_DEVICE_TIM1_CH4: +#ifdef USE_MOTOR_DSHOT + dshot_dma_isr(ass); +#endif + break; -static void handle_dma_stream_isr(dma_device_t dev) { - switch (dev) { case DMA_DEVICE_SPI1_RX: case DMA_DEVICE_SPI2_RX: case DMA_DEVICE_SPI3_RX: @@ -190,25 +221,66 @@ static void handle_dma_stream_isr(dma_device_t dev) { case DMA_DEVICE_SPI2_TX: case DMA_DEVICE_SPI3_TX: case DMA_DEVICE_SPI4_TX: - spi_dma_isr(dev); + spi_dma_isr(ass); break; - case DMA_DEVICE_TIM1_CH1: - case DMA_DEVICE_TIM1_CH3: - case DMA_DEVICE_TIM1_CH4: -#ifdef USE_MOTOR_DSHOT - dshot_dma_isr(dev); + + case DMA_DEVICE_RGB: +#ifdef USE_RGB_LED + rgb_dma_isr(ass); #endif break; + case DMA_DEVICE_MAX: break; } } -#define DMA_STREAM(_port, _chan, _stream, _dev) \ - void DMA##_port##_Stream##_stream##_IRQHandler() { \ - handle_dma_stream_isr(DMA_DEVICE_##_dev); \ - } - -DMA_STREAMS +void DMA1_Stream0_IRQHandler() { + handle_dma_stream_isr(&dma_assigments[0]); +} +void DMA1_Stream1_IRQHandler() { + handle_dma_stream_isr(&dma_assigments[1]); +} +void DMA1_Stream2_IRQHandler() { + handle_dma_stream_isr(&dma_assigments[2]); +} +void DMA1_Stream3_IRQHandler() { + handle_dma_stream_isr(&dma_assigments[3]); +} +void DMA1_Stream4_IRQHandler() { + handle_dma_stream_isr(&dma_assigments[4]); +} +void DMA1_Stream5_IRQHandler() { + handle_dma_stream_isr(&dma_assigments[5]); +} +void DMA1_Stream6_IRQHandler() { + handle_dma_stream_isr(&dma_assigments[6]); +} +void DMA1_Stream7_IRQHandler() { + handle_dma_stream_isr(&dma_assigments[7]); +} -#undef DMA_STREAM \ No newline at end of file +void DMA2_Stream0_IRQHandler() { + handle_dma_stream_isr(&dma_assigments[8]); +} +void DMA2_Stream1_IRQHandler() { + handle_dma_stream_isr(&dma_assigments[9]); +} +void DMA2_Stream2_IRQHandler() { + handle_dma_stream_isr(&dma_assigments[10]); +} +void DMA2_Stream3_IRQHandler() { + handle_dma_stream_isr(&dma_assigments[11]); +} +void DMA2_Stream4_IRQHandler() { + handle_dma_stream_isr(&dma_assigments[12]); +} +void DMA2_Stream5_IRQHandler() { + handle_dma_stream_isr(&dma_assigments[13]); +} +void DMA2_Stream6_IRQHandler() { + handle_dma_stream_isr(&dma_assigments[14]); +} +void DMA2_Stream7_IRQHandler() { + handle_dma_stream_isr(&dma_assigments[15]); +} \ No newline at end of file diff --git a/src/driver/stm32/dma.h b/src/driver/stm32/dma.h new file mode 100644 index 000000000..c0703e649 --- /dev/null +++ b/src/driver/stm32/dma.h @@ -0,0 +1,40 @@ +#pragma once + +#include "hardware.h" + +#include "driver/resource.h" + +typedef enum { + DMA_DEVICE_INVALID, + + DMA_DEVICE_SPI1_RX, + DMA_DEVICE_SPI1_TX, + DMA_DEVICE_SPI2_RX, + DMA_DEVICE_SPI2_TX, + DMA_DEVICE_SPI3_RX, + DMA_DEVICE_SPI3_TX, + DMA_DEVICE_SPI4_RX, + DMA_DEVICE_SPI4_TX, + DMA_DEVICE_TIM1_CH1, + DMA_DEVICE_TIM1_CH3, + DMA_DEVICE_TIM1_CH4, + DMA_DEVICE_RGB, + + DMA_DEVICE_MAX, +} dma_device_t; + +typedef struct { + DMA_TypeDef *port; + uint8_t port_index; + + DMA_Stream_TypeDef *stream; + uint8_t stream_index; + + IRQn_Type irq; +} dma_stream_def_t; + +typedef struct { + dma_device_t dev; + const dma_stream_def_t *def; + uint32_t channel; +} dma_assigment_t; \ No newline at end of file diff --git a/src/driver/stm32/hardware.h b/src/driver/stm32/hardware.h new file mode 100644 index 000000000..f7b777b96 --- /dev/null +++ b/src/driver/stm32/hardware.h @@ -0,0 +1,87 @@ +#pragma once + +#ifdef STM32F4 +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#endif + +#ifdef STM32F7 +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#endif + +#ifdef STM32H7 +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#endif + +#ifdef STM32F411 +#define SYS_CLOCK_FREQ_HZ 108000000 +#define PWM_CLOCK_FREQ_HZ 108000000 +#define SPI_CLOCK_FREQ_HZ (SYS_CLOCK_FREQ_HZ / 2) + +#define LOOPTIME LOOPTIME_4K +#endif + +#ifdef STM32F405 +#define SYS_CLOCK_FREQ_HZ 168000000 +#define PWM_CLOCK_FREQ_HZ 84000000 +#define SPI_CLOCK_FREQ_HZ (SYS_CLOCK_FREQ_HZ / 4) + +#define LOOPTIME LOOPTIME_8K +#endif + +#ifdef STM32F7 +#define SYS_CLOCK_FREQ_HZ 216000000 +#define PWM_CLOCK_FREQ_HZ 216000000 +#define SPI_CLOCK_FREQ_HZ (SYS_CLOCK_FREQ_HZ / 4) + +#define LOOPTIME LOOPTIME_8K + +#define WITHIN_DTCM_RAM(p) (((uint32_t)p & 0xffff0000) == 0x20000000) +#define WITHIN_DMA_RAM(p) (false) +#endif + +#ifdef STM32H7 +#define SYS_CLOCK_FREQ_HZ 480000000 +#define PWM_CLOCK_FREQ_HZ (SYS_CLOCK_FREQ_HZ / 2) +#define SPI_CLOCK_FREQ_HZ (SYS_CLOCK_FREQ_HZ / 4) + +#define LOOPTIME LOOPTIME_8K + +#define WITHIN_DTCM_RAM(p) (((uint32_t)p & 0xfffe0000) == 0x20000000) +#define WITHIN_DMA_RAM(p) (((uint32_t)p & 0xfffe0000) == 0x30000000) +#endif \ No newline at end of file diff --git a/src/driver/stm32/system.h b/src/driver/stm32/system.h index 8c135d85d..ed10a0891 100644 --- a/src/driver/stm32/system.h +++ b/src/driver/stm32/system.h @@ -1,112 +1,12 @@ #pragma once -#ifdef STM32F4 -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#endif - -#ifdef STM32F7 -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#endif - -#ifdef STM32H7 -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#endif - -#ifdef STM32F411 -#define SYS_CLOCK_FREQ_HZ 108000000 -#define PWM_CLOCK_FREQ_HZ 108000000 -#define SPI_CLOCK_FREQ_HZ (SYS_CLOCK_FREQ_HZ / 2) - -#define LOOPTIME LOOPTIME_4K -#endif - -#ifdef STM32F405 -#define SYS_CLOCK_FREQ_HZ 168000000 -#define PWM_CLOCK_FREQ_HZ 84000000 -#define SPI_CLOCK_FREQ_HZ (SYS_CLOCK_FREQ_HZ / 4) - -#define LOOPTIME LOOPTIME_8K -#endif - -#ifdef STM32F7 -#define SYS_CLOCK_FREQ_HZ 216000000 -#define PWM_CLOCK_FREQ_HZ 216000000 -#define SPI_CLOCK_FREQ_HZ (SYS_CLOCK_FREQ_HZ / 4) - -#define LOOPTIME LOOPTIME_8K - -#define WITHIN_DTCM_RAM(p) (((uint32_t)p & 0xffff0000) == 0x20000000) -#define WITHIN_DMA_RAM(p) (false) -#endif - -#ifdef STM32H7 -#define SYS_CLOCK_FREQ_HZ 480000000 -#define PWM_CLOCK_FREQ_HZ (SYS_CLOCK_FREQ_HZ / 2) -#define SPI_CLOCK_FREQ_HZ (SYS_CLOCK_FREQ_HZ / 4) - -#define LOOPTIME LOOPTIME_8K - -#define WITHIN_DTCM_RAM(p) (((uint32_t)p & 0xfffe0000) == 0x20000000) -#define WITHIN_DMA_RAM(p) (((uint32_t)p & 0xfffe0000) == 0x30000000) -#endif +#include "hardware.h" #include "adc.h" +#include "dma.h" #include "rcc.h" typedef GPIO_TypeDef gpio_port_t; typedef SPI_TypeDef spi_port_t; typedef TIM_TypeDef timer_dev_t; typedef USART_TypeDef usart_dev_t; - -typedef struct { - uint32_t device; - - DMA_TypeDef *port; - uint8_t port_index; - - uint32_t channel; - uint8_t channel_index; - - uint32_t request; - - DMA_Stream_TypeDef *stream; - uint8_t stream_index; - - IRQn_Type irq; -} dma_stream_def_t; \ No newline at end of file diff --git a/src/driver/timer.h b/src/driver/timer.h index 9a704c8ef..92cbb075e 100644 --- a/src/driver/timer.h +++ b/src/driver/timer.h @@ -51,6 +51,7 @@ typedef enum { TIMER_USE_FREE, TIMER_USE_MOTOR_DSHOT, TIMER_USE_MOTOR_PWM, + TIMER_USE_RGB_LED, TIMER_USE_ELRS, TIMER_USE_SOFT_SERIAL, } timer_use_t; @@ -72,5 +73,6 @@ void timer_alloc_init(); bool timer_alloc_tag(timer_use_t use, resource_tag_t tag); resource_tag_t timer_alloc(timer_use_t use); uint32_t timer_channel_val(timer_channel_t chan); +void timer_enable_dma_request(timer_index_t tim, timer_channel_t chan, bool state); void timer_up_init(timer_index_t tim, uint16_t divider, uint32_t period); \ No newline at end of file diff --git a/src/io/rgb_led.c b/src/io/rgb_led.c index 59306ee67..f4a792942 100644 --- a/src/io/rgb_led.c +++ b/src/io/rgb_led.c @@ -1,168 +1,17 @@ -#include +#include "io/rgb_led.h" #include "core/project.h" #include "driver/rgb_led.h" -#include "driver/time.h" -#include "flight/control.h" -#include "util/util.h" -// normal flight rgb colour - LED switch ON -#define RGB_VALUE_INFLIGHT_ON RGB(255, 255, 255) +static const uint32_t green = RGB(98, 168, 52); -// normal flight rgb colour - LED switch OFF -#define RGB_VALUE_INFLIGHT_OFF RGB(0, 0, 0) - -// colour before bind -#define RGB_VALUE_BEFORE_BIND RGB(0, 128, 128) - -// fade from one color to another when changed -#define RGB_FILTER_ENABLE -#define RGB_FILTER_TIME_MICROSECONDS 50e3 - -// runs the update once every 16 loop times ( 16 mS ) -#define DOWNSAMPLE 16 - -#define RGB_FILTER_TIME FILTERCALC(1000 * DOWNSAMPLE, RGB_FILTER_TIME_MICROSECONDS) -#define RGB(r, g, b) ((((int)g & 0xff) << 16) | (((int)r & 0xff) << 8) | ((int)b & 0xff)) - -#if (RGB_LED_NUMBER > 0) - -// array with individual led brightnesses -int rgb_led_value[RGB_LED_NUMBER]; -// loop count for downsampling -int rgb_loopcount = 0; -// rgb low pass filter variables -float r_filt, g_filt, b_filt; - -// sets all leds to a brightness -void rgb_led_set_all(int rgb) { -#ifdef RGB_FILTER_ENABLE - // deconstruct the colour into components - int g = rgb >> 16; - int r = (rgb & 0x0000FF00) >> 8; - int b = rgb & 0xff; - - // filter individual colors - lpf(&r_filt, r, RGB_FILTER_TIME); - lpf(&g_filt, g, RGB_FILTER_TIME); - lpf(&b_filt, b, RGB_FILTER_TIME); - - int temp = RGB(r_filt, g_filt, b_filt); - - for (int i = 0; i < RGB_LED_NUMBER; i++) - rgb_led_value[i] = temp; - -#else - for (int i = 0; i < RGB_LED_NUMBER; i++) - rgb_led_value[i] = rgb; -#endif -} - -// set an individual led brightness -void rgb_led_set_one(int led_number, int rgb) { - rgb_led_value[led_number] = rgb; -} - -// flashes between 2 colours, duty cycle 1 - 15 -void rgb_ledflash(int color1, int color2, uint32_t period, int duty) { - if (time_micros() % period > (period * duty) >> 4) { - rgb_led_set_all(color1); - } else { - rgb_led_set_all(color2); - } -} - -// speed of movement -float KR_SPEED = 0.005f * DOWNSAMPLE; - -float kr_position = 0; -int kr_dir = 0; - -// knight rider style led movement -void rgb_knight_rider() { - if (kr_dir) { - kr_position += KR_SPEED; - if (kr_position > RGB_LED_NUMBER - 1) - kr_dir = !kr_dir; - } else { - kr_position -= KR_SPEED; - if (kr_position < 0) - kr_dir = !kr_dir; +void rgb_led_update() { +#if defined(USE_RGB_LED) + if (rgb_led_busy()) { + return; } - // calculate led value - for (int i = 0; i < RGB_LED_NUMBER; i++) { - float led_bright = fabsf((float)i - kr_position); - if (led_bright > 1.0f) - led_bright = 1.0f; - led_bright = 1.0f - led_bright; - - // set a green background as well, 32 brightness - rgb_led_set_one(i, RGB((led_bright * 255.0f), (32.0f - led_bright * 32.0f), 0)); - } -} - -// 2 led flasher -void rgb_ledflash_twin(int color1, int color2, uint32_t period) { - if (time_micros() % period > (period / 2)) { - for (int i = 0; i < RGB_LED_NUMBER; i++) { - if ((i / 2) * 2 == i) - rgb_led_set_one(i, color1); - else - rgb_led_set_one(i, color2); - } - } else { - for (int i = 0; i < RGB_LED_NUMBER; i++) { - if ((i / 2) * 2 == i) - rgb_led_set_one(i, color2); - else - rgb_led_set_one(i, color1); - } - } -} - -// main function -void rgb_led_lvc() { - rgb_loopcount++; - if (rgb_loopcount > DOWNSAMPLE) { - rgb_loopcount = 0; - // led flash logic - if (flags.lowbatt) { - // rgb_led_set_all( RGB( 255 , 0 , 0 ) ); - rgb_ledflash(RGB(255, 0, 0), RGB(255, 32, 0), 500000, 8); - } else { - if (flags.rx_mode == RXMODE_BIND) { - // bind mode - // rgb_ledflash ( RGB( 0 , 0 , 255 ), RGB( 0 , 128 , 0 ), 1000000, 12); - // rgb_ledflash_twin( RGB( 0 , 0 , 255 ), RGB( 0 , 128 , 0 ), 1000000); - rgb_led_set_all(RGB_VALUE_BEFORE_BIND); - // rgb_knight_rider(); - } else { // non bind - if (flags.failsafe) { - // failsafe flash - rgb_ledflash(RGB(0, 128, 0), RGB(0, 0, 128), 500000, 8); - // rgb_led_set_all( RGB( 0 , 128 , 128 ) ); - } else { - - if (rx_aux_on(AUX_LEDS_ON)) - - rgb_led_set_all(RGB_VALUE_INFLIGHT_ON); - else - rgb_led_set_all(RGB_VALUE_INFLIGHT_OFF); - } - } - } - -#ifdef RGB_LED_DMA - // send dma start signal - rgb_send(0); -#else - // send data to leds - for (int i = 0; i < RGB_LED_NUMBER; i++) { - rgb_send(rgb_led_value[i]); - } + rgb_led_set_value(green, 2); + rgb_led_send(); #endif - } } - -#endif diff --git a/src/io/rgb_led.h b/src/io/rgb_led.h index 686b389ac..921d3fee2 100644 --- a/src/io/rgb_led.h +++ b/src/io/rgb_led.h @@ -1,3 +1,5 @@ #pragma once -void rgb_led_lvc(); \ No newline at end of file +#define RGB(r, g, b) ((((uint32_t)g & 0xff) << 16) | (((uint32_t)r & 0xff) << 8) | ((uint32_t)b & 0xff)) + +void rgb_led_update(); \ No newline at end of file