Add ADC Handling. Write first draft of operating modes for testing
This commit is contained in:
parent
d3b1c8fc3f
commit
6104ccd08a
1
Makefile
1
Makefile
@ -14,6 +14,7 @@ endif
|
||||
CFILES := main.c setup/system_init.c startup/startup_stm32f0xx.c systick.c
|
||||
CFILES += i2c.c
|
||||
CFILES += dmx.c
|
||||
CFILES += poti.c
|
||||
ASFILES =
|
||||
INCLUDEPATH = -Iinclude -Iinclude/cmsis
|
||||
|
||||
|
16
dmx.c
16
dmx.c
@ -12,15 +12,6 @@ static uint8_t *dmx_data_ptr;
|
||||
static uint16_t dmx_tx_break_len;
|
||||
static uint16_t dmx_tx_break_pause;
|
||||
|
||||
enum dmx_tx_state {
|
||||
DMX_TX_OFF,
|
||||
DMX_TX_IDLE,
|
||||
DMX_TX_BREAK,
|
||||
DMX_TX_BREAK_PAUSE,
|
||||
DMX_TX_NULLBYTE,
|
||||
DMX_TX_DATA,
|
||||
};
|
||||
|
||||
static volatile enum dmx_tx_state tx_state;
|
||||
|
||||
/* USART1 TX is mapped on DMA Channel2 */
|
||||
@ -147,6 +138,11 @@ void TIM14_IRQHandler(void)
|
||||
}
|
||||
}
|
||||
|
||||
enum dmx_tx_state dmx_tx_get_current_state(void)
|
||||
{
|
||||
return tx_state;
|
||||
}
|
||||
|
||||
void USART1_IRQHandler(void)
|
||||
{
|
||||
uint32_t isr;
|
||||
@ -175,7 +171,7 @@ void DMA_CH2_3_DMA2_CH1_2_IRQHandler(void)
|
||||
uint32_t isr;
|
||||
|
||||
isr = DMA1->ISR;
|
||||
DMA1->IFCR = isr;
|
||||
DMA1->IFCR = isr & DMA_ISR_TCIF2;
|
||||
|
||||
if (isr & DMA_ISR_TCIF2) {
|
||||
/* Channnel 2 trasnfer complete. Universe trasnferred */
|
||||
|
@ -3,6 +3,14 @@
|
||||
|
||||
#include <stdint.h>
|
||||
|
||||
enum dmx_tx_state {
|
||||
DMX_TX_OFF,
|
||||
DMX_TX_IDLE,
|
||||
DMX_TX_BREAK,
|
||||
DMX_TX_BREAK_PAUSE,
|
||||
DMX_TX_NULLBYTE,
|
||||
DMX_TX_DATA,
|
||||
};
|
||||
|
||||
void dmx_init(uint8_t *data, uint32_t universe_length, GPIO_TypeDef *tx_port, uint8_t tx_pin,
|
||||
uint16_t dmx_delay, uint16_t dmx_break_len, uint16_t break_pause);
|
||||
@ -11,4 +19,6 @@ void dmx_stream_start(void);
|
||||
|
||||
void dmx_stream_stop(void);
|
||||
|
||||
enum dmx_tx_state dmx_tx_get_current_state(void);
|
||||
|
||||
#endif /* _DMX_H_ */
|
19
include/poti.h
Normal file
19
include/poti.h
Normal file
@ -0,0 +1,19 @@
|
||||
#ifndef _POTI_H_
|
||||
#define _POTI_H_
|
||||
|
||||
#include <stdint.h>
|
||||
|
||||
#define POTI_COUNT (3)
|
||||
|
||||
#define POTI_ADC_CHANNELS 0, 1, 4
|
||||
|
||||
struct poti_values {
|
||||
uint16_t current_pot_vals[POTI_COUNT];
|
||||
uint8_t pot_vals_filtered[POTI_COUNT];
|
||||
};
|
||||
|
||||
void poti_init_adc(void);
|
||||
|
||||
void poti_get_values(struct poti_values *out_val);
|
||||
|
||||
#endif /* _POTI_H_ */
|
111
main.c
111
main.c
@ -1,3 +1,4 @@
|
||||
#include <poti.h>
|
||||
#include <i2c.h>
|
||||
#include <stdint.h>
|
||||
#include <stm32f0xx.h>
|
||||
@ -15,8 +16,36 @@
|
||||
#define PCA9555_REG_CONFIG0 (0x6)
|
||||
#define PCA9555_REG_CONFIG1 (0x7)
|
||||
|
||||
|
||||
static const uint8_t gamma8[] = {
|
||||
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1,
|
||||
1, 1, 1, 1, 1, 1, 1, 1, 1, 2, 2, 2, 2, 2, 2, 2,
|
||||
2, 3, 3, 3, 3, 3, 3, 3, 4, 4, 4, 4, 4, 5, 5, 5,
|
||||
5, 6, 6, 6, 6, 7, 7, 7, 7, 8, 8, 8, 9, 9, 9, 10,
|
||||
10, 10, 11, 11, 11, 12, 12, 13, 13, 13, 14, 14, 15, 15, 16, 16,
|
||||
17, 17, 18, 18, 19, 19, 20, 20, 21, 21, 22, 22, 23, 24, 24, 25,
|
||||
25, 26, 27, 27, 28, 29, 29, 30, 31, 32, 32, 33, 34, 35, 35, 36,
|
||||
37, 38, 39, 39, 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 50,
|
||||
51, 52, 54, 55, 56, 57, 58, 59, 60, 61, 62, 63, 64, 66, 67, 68,
|
||||
69, 70, 72, 73, 74, 75, 77, 78, 79, 81, 82, 83, 85, 86, 87, 89,
|
||||
90, 92, 93, 95, 96, 98, 99,101,102,104,105,107,109,110,112,114,
|
||||
115,117,119,120,122,124,126,127,129,131,133,135,137,138,140,142,
|
||||
144,146,148,150,152,154,156,158,160,162,164,167,169,171,173,175,
|
||||
177,180,182,184,186,189,191,193,196,198,200,203,205,208,210,213,
|
||||
215,218,220,223,225,228,231,233,236,239,241,244,247,249,252,255
|
||||
};
|
||||
|
||||
static uint8_t dmx_universe[129];
|
||||
|
||||
enum color_mode {
|
||||
MODE_RED,
|
||||
MODE_GREEN,
|
||||
MODE_BLUE,
|
||||
MODE_RGB,
|
||||
MODE_WHITE_DISCRETE,
|
||||
};
|
||||
|
||||
static void setup_pins(void)
|
||||
{
|
||||
uint32_t tmp;
|
||||
@ -26,8 +55,8 @@ static void setup_pins(void)
|
||||
tmp = GPIOA->MODER;
|
||||
|
||||
/* Reset all used port pins */
|
||||
tmp &= MODER_DELETE(0) & MODER_DELETE(1) & MODER_DELETE(2) & MODER_DELETE(3) & \
|
||||
MODER_DELETE(4) & MODER_DELETE(5) & MODER_DELETE(6) & MODER_DELETE(7) & \
|
||||
tmp &= MODER_DELETE(0) & MODER_DELETE(1) & MODER_DELETE(2) & MODER_DELETE(3) &
|
||||
MODER_DELETE(4) & MODER_DELETE(5) & MODER_DELETE(6) & MODER_DELETE(7) &
|
||||
MODER_DELETE(8) & MODER_DELETE(9) & MODER_DELETE(10);
|
||||
|
||||
/* Analog ports for POTIs */
|
||||
@ -58,24 +87,18 @@ static void setup_pins(void)
|
||||
GPIOB->PUPDR = PULLUP(1);
|
||||
}
|
||||
|
||||
static void zero_dmx_universe(void)
|
||||
{
|
||||
int i;
|
||||
|
||||
for (i = 0; i < sizeof(dmx_universe); i++) {
|
||||
dmx_universe[i] = 0;
|
||||
}
|
||||
}
|
||||
|
||||
int main(void)
|
||||
{
|
||||
uint8_t i2c_command[2];
|
||||
uint8_t port;
|
||||
uint32_t odr;
|
||||
enum color_mode mode = MODE_RED;
|
||||
struct poti_values poti_vals;
|
||||
|
||||
setup_pins();
|
||||
i2c_init();
|
||||
dmx_init(dmx_universe, sizeof(dmx_universe), GPIOA, 2u, 2300u, 10u, 5u);
|
||||
poti_init_adc();
|
||||
|
||||
/* Setup Systick for 1ms ticks */
|
||||
SysTick_Config(48000000UL/1000);
|
||||
@ -98,33 +121,69 @@ int main(void)
|
||||
|
||||
/* Blink the LEDs */
|
||||
while (1) {
|
||||
systick_wait_ms(3);
|
||||
systick_wait_ms(10);
|
||||
poti_get_values(&poti_vals);
|
||||
i2c_read(PCA9555_ADDR, PCA9555_REG_IN_PORT0, &port, 1u);
|
||||
i2c_command[0] = PCA9555_REG_OUT_PORT1;
|
||||
i2c_command[1] = ~port;
|
||||
i2c_write(PCA9555_ADDR, i2c_command, 2u);
|
||||
odr = GPIOA->ODR;
|
||||
odr &= ~((1<<5) | (1<<6) | (1<<7));
|
||||
if (port & (1<<0)) {
|
||||
zero_dmx_universe();
|
||||
for (int i = 0; i < 32; i++) {
|
||||
dmx_universe[i*4] = 0xFF;
|
||||
}
|
||||
mode = MODE_RED;
|
||||
}
|
||||
if (port & (1<<1)) {
|
||||
zero_dmx_universe();
|
||||
for (int i = 0; i < 32; i++) {
|
||||
dmx_universe[i*4+1] = 0xFF;
|
||||
}
|
||||
mode = MODE_GREEN;
|
||||
}
|
||||
if (port & (1<<2)) {
|
||||
zero_dmx_universe();
|
||||
for (int i = 0; i < 32; i++) {
|
||||
dmx_universe[i*4+2] = 0xFF;
|
||||
}
|
||||
mode = MODE_BLUE;
|
||||
}
|
||||
if (port & (1<<3)) {
|
||||
mode = MODE_RGB;
|
||||
}
|
||||
if (port & (1<<4)) {
|
||||
mode = MODE_WHITE_DISCRETE;
|
||||
}
|
||||
GPIOA->ODR = odr;
|
||||
|
||||
i2c_write(PCA9555_ADDR, i2c_command, 2u);
|
||||
|
||||
switch (mode) {
|
||||
case MODE_RED:
|
||||
case MODE_GREEN:
|
||||
case MODE_BLUE:
|
||||
for (int i = 0; i < 32; i++) {
|
||||
dmx_universe[i*4] = mode == MODE_RED ? poti_vals.pot_vals_filtered[0] : 0;
|
||||
dmx_universe[i*4+1] = mode == MODE_GREEN ? poti_vals.pot_vals_filtered[0] : 0;
|
||||
dmx_universe[i*4+2] = mode == MODE_BLUE ? poti_vals.pot_vals_filtered[0] : 0;
|
||||
dmx_universe[i*4+3] = 0x0;
|
||||
}
|
||||
dmx_universe[128] = 0;
|
||||
odr |= (1<<5);
|
||||
break;
|
||||
case MODE_RGB:
|
||||
odr |= (1<<5) | (1<<6) | (1<<7);
|
||||
for (int i = 0; i < 32; i++) {
|
||||
dmx_universe[i*4] =poti_vals.pot_vals_filtered[0];
|
||||
dmx_universe[i*4+1] = poti_vals.pot_vals_filtered[1];
|
||||
dmx_universe[i*4+2] = poti_vals.pot_vals_filtered[2];
|
||||
dmx_universe[i*4+3] = 0x0;
|
||||
}
|
||||
dmx_universe[128] = 0;
|
||||
break;
|
||||
case MODE_WHITE_DISCRETE:
|
||||
odr |= (1<<5);
|
||||
for (int i = 0; i < 32; i++) {
|
||||
dmx_universe[i*4] = 0;
|
||||
dmx_universe[i*4+1] = 0;
|
||||
dmx_universe[i*4+2] = 0;
|
||||
dmx_universe[i*4+3] = 0;
|
||||
}
|
||||
dmx_universe[128] = gamma8[poti_vals.pot_vals_filtered[0]];
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
|
||||
GPIOA->ODR = odr;
|
||||
|
||||
}
|
||||
}
|
||||
|
122
poti.c
Normal file
122
poti.c
Normal file
@ -0,0 +1,122 @@
|
||||
#include "stm32f030x6.h"
|
||||
#include <poti.h>
|
||||
#include <stdint.h>
|
||||
|
||||
const uint8_t adc_channels[POTI_COUNT] = {POTI_ADC_CHANNELS};
|
||||
|
||||
static volatile uint16_t adc_raw_values[POTI_COUNT];
|
||||
|
||||
static volatile uint16_t adc_moving_avg[POTI_COUNT];
|
||||
|
||||
static volatile uint8_t adc_moving_avg_out[POTI_COUNT];
|
||||
|
||||
/**
|
||||
* @brief Perform calibration. ADC must be disabled
|
||||
*/
|
||||
static void perform_cal(void)
|
||||
{
|
||||
ADC1->CR = ADC_CR_ADCAL;
|
||||
while (ADC1->CR & ADC_CR_ADCAL);
|
||||
(void)ADC1->DR;
|
||||
}
|
||||
|
||||
void poti_init_adc(void)
|
||||
{
|
||||
int i;
|
||||
uint32_t reg;
|
||||
|
||||
RCC->APB2ENR |= RCC_APB2ENR_ADCEN;
|
||||
RCC->AHBENR |= RCC_AHBENR_DMAEN;
|
||||
|
||||
for (i = 0; i < POTI_COUNT; i++) {
|
||||
adc_moving_avg[i] = 0u;
|
||||
}
|
||||
|
||||
DMA1_Channel1->CCR = 0ul;
|
||||
|
||||
/* Clear ADC ready bit */
|
||||
ADC1->ISR = ADC_ISR_ADRDY;
|
||||
|
||||
ADC1->CFGR1 = 0;
|
||||
|
||||
/* Start calibration */
|
||||
perform_cal();
|
||||
|
||||
/* Activate channels to convert */
|
||||
reg = 0ul;
|
||||
for (i = 0; i < POTI_COUNT; i++) {
|
||||
if (adc_channels[i] <= 17)
|
||||
reg |= (1<<adc_channels[i]);
|
||||
}
|
||||
ADC1->CHSELR = reg;
|
||||
|
||||
/* Setup sample time (71.5 cycles) */
|
||||
ADC1->SMPR = ADC_SMPR_SMP_2 | ADC_SMPR_SMP_1;
|
||||
|
||||
/* Setup ADC clock (PCLK / 4) */
|
||||
ADC1->CFGR2 = ADC_CFGR2_CKMODE_1;
|
||||
|
||||
/* Setup continuous sample mode with DMA in circular mode */
|
||||
ADC1->CFGR1 = ADC_CFGR1_CONT | ADC_CFGR1_DMAEN | ADC_CFGR1_DMACFG;
|
||||
|
||||
/* Enable ADC and wait for ready */
|
||||
ADC1->CR = ADC_CR_ADEN;
|
||||
while (!(ADC1->ISR & ADC_ISR_ADRDY));
|
||||
|
||||
/* Setup DMA channel 1 */
|
||||
DMA1_Channel1->CPAR = (uint32_t)&ADC1->DR;
|
||||
DMA1_Channel1->CMAR = (uint32_t)adc_raw_values;
|
||||
|
||||
DMA1_Channel1->CNDTR = POTI_COUNT;
|
||||
|
||||
DMA1_Channel1->CCR = DMA_CCR_PL_0 | DMA_CCR_MSIZE_0 | DMA_CCR_PSIZE_0 | DMA_CCR_MINC |
|
||||
DMA_CCR_CIRC | DMA_CCR_TCIE | DMA_CCR_EN;
|
||||
|
||||
|
||||
NVIC_EnableIRQ(DMA1_Channel1_IRQn);
|
||||
|
||||
/* Start the conversion */
|
||||
ADC1->CR |= ADC_CR_ADSTART;
|
||||
}
|
||||
|
||||
void poti_get_values(struct poti_values *out_val)
|
||||
{
|
||||
int i;
|
||||
|
||||
if (!out_val)
|
||||
return;
|
||||
|
||||
for (i = 0; i < POTI_COUNT; i++) {
|
||||
out_val->current_pot_vals[i] = adc_raw_values[i];
|
||||
out_val->pot_vals_filtered[i] = adc_moving_avg_out[i];
|
||||
}
|
||||
}
|
||||
|
||||
static uint16_t calc_moving_avg(uint16_t adc_val, uint16_t prev_avg)
|
||||
{
|
||||
uint32_t val;
|
||||
|
||||
val = (uint32_t)prev_avg;
|
||||
val /= 16u;
|
||||
val *= 15u;
|
||||
val += adc_val;
|
||||
|
||||
return val;
|
||||
}
|
||||
|
||||
void DMA_CH1_IRQHandler(void)
|
||||
{
|
||||
uint32_t isr;
|
||||
int i;
|
||||
|
||||
isr = DMA1->ISR;
|
||||
DMA1->IFCR = isr & DMA_ISR_TCIF1;
|
||||
|
||||
if (isr & DMA_ISR_TCIF1) {
|
||||
/* All ADC samples converted */
|
||||
for (i = 0; i < POTI_COUNT; i++) {
|
||||
adc_moving_avg[i] = calc_moving_avg(adc_raw_values[i], adc_moving_avg[i]);
|
||||
adc_moving_avg_out[i] = (uint8_t)(adc_moving_avg[i] / 16u / 16u);
|
||||
}
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue
Block a user