2024-05-01 21:43:05 +08:00
|
|
|
#include "stm32f4xx_rcc.h"
|
|
|
|
|
#include "stm32f4xx_gpio.h"
|
|
|
|
|
#include "stm32f4xx_tim.h"
|
|
|
|
|
#include "stm32f4xx_spi.h"
|
|
|
|
|
#include "ltc1867.h"
|
|
|
|
|
#include "debug.h"
|
|
|
|
|
|
2024-06-29 15:22:55 +08:00
|
|
|
__attribute__((section(".bss_ccm"))) uint32_t ltc1867_samp_cnt = 0;
|
|
|
|
|
__attribute__((section(".data_ccm"))) uint16_t ltc1867_spi_data[LTC1867_CH_CNT] = {0x9400, 0x8400}; /* order: ch2, ch0 */
|
|
|
|
|
|
2024-05-01 21:43:05 +08:00
|
|
|
void ltc1867_init(void)
|
|
|
|
|
{
|
|
|
|
|
GPIO_InitTypeDef GPIO_InitStructure;
|
|
|
|
|
NVIC_InitTypeDef NVIC_InitStructure;
|
|
|
|
|
TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
|
|
|
|
|
SPI_InitTypeDef SPI_InitStructure;
|
|
|
|
|
|
2024-06-29 15:22:55 +08:00
|
|
|
/* TIM2 32bit for sample */
|
|
|
|
|
RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE);
|
|
|
|
|
NVIC_InitStructure.NVIC_IRQChannel = TIM2_IRQn;
|
2024-05-01 21:43:05 +08:00
|
|
|
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
|
|
|
|
|
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
|
|
|
|
|
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
|
|
|
|
|
NVIC_Init(&NVIC_InitStructure);
|
|
|
|
|
/* Time base configuration */
|
2024-06-29 15:22:55 +08:00
|
|
|
TIM_TimeBaseStructure.TIM_Period = TIMER_PERIOD_SAMPLE;
|
|
|
|
|
TIM_TimeBaseStructure.TIM_Prescaler = 0;
|
2024-05-01 21:43:05 +08:00
|
|
|
TIM_TimeBaseStructure.TIM_ClockDivision = 0;
|
|
|
|
|
TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;
|
2024-06-29 15:22:55 +08:00
|
|
|
TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure);
|
|
|
|
|
TIM_PrescalerConfig(TIM2, 0, TIM_PSCReloadMode_Immediate);
|
|
|
|
|
TIM_ITConfig(TIM2, TIM_IT_Update, ENABLE);
|
|
|
|
|
TIM_Cmd(TIM2, ENABLE);
|
2024-05-01 21:43:05 +08:00
|
|
|
|
2024-06-29 15:22:55 +08:00
|
|
|
/* TIM3 16bit for ADC conversion */
|
|
|
|
|
RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE);
|
2024-05-01 21:43:05 +08:00
|
|
|
TIM_TimeBaseStructure.TIM_Period = TIMER_PERIOD_DELAY;
|
|
|
|
|
TIM_TimeBaseStructure.TIM_Prescaler = 0;
|
|
|
|
|
TIM_TimeBaseStructure.TIM_ClockDivision = 0;
|
|
|
|
|
TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;
|
2024-06-29 15:22:55 +08:00
|
|
|
TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure);
|
|
|
|
|
TIM_PrescalerConfig(TIM3, 0, TIM_PSCReloadMode_Immediate);
|
|
|
|
|
TIM_Cmd(TIM3, DISABLE);
|
2024-05-01 21:43:05 +08:00
|
|
|
|
|
|
|
|
/* LEDIR pin configure */
|
|
|
|
|
GPIO_InitStructure.GPIO_Pin = PIN_LEDIR;
|
2024-06-29 15:22:55 +08:00
|
|
|
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT;
|
2024-05-01 21:43:05 +08:00
|
|
|
GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
|
2024-06-29 15:22:55 +08:00
|
|
|
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_25MHz;
|
|
|
|
|
GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;
|
2024-05-01 21:43:05 +08:00
|
|
|
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOB, ENABLE);
|
|
|
|
|
GPIO_Init(PORT_LEDIR, &GPIO_InitStructure);
|
2024-06-29 15:22:55 +08:00
|
|
|
ltc1867_ledir_off();
|
2024-05-01 21:43:05 +08:00
|
|
|
|
|
|
|
|
/* LTC1867 conv pin configure */
|
|
|
|
|
GPIO_InitStructure.GPIO_Pin = PIN_CONV;
|
|
|
|
|
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT;
|
|
|
|
|
GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
|
2024-06-29 15:22:55 +08:00
|
|
|
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_25MHz;
|
|
|
|
|
GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;
|
2024-05-01 21:43:05 +08:00
|
|
|
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE);
|
|
|
|
|
GPIO_Init(PORT_LTC1867, &GPIO_InitStructure);
|
|
|
|
|
/* config SPI for LTC1867 communication */
|
|
|
|
|
GPIO_PinAFConfig(PORT_SPI, GPIO_PinSource5, GPIO_AF_SPI1);
|
2024-06-29 15:22:55 +08:00
|
|
|
GPIO_PinAFConfig(PORT_SPI, GPIO_PinSource6, GPIO_AF_SPI1);
|
2024-05-01 21:43:05 +08:00
|
|
|
GPIO_PinAFConfig(PORT_SPI, GPIO_PinSource7, GPIO_AF_SPI1);
|
|
|
|
|
GPIO_InitStructure.GPIO_Pin = PIN_SPI_SCK | PIN_SPI_SDI | PIN_SPI_SDO;
|
|
|
|
|
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
|
|
|
|
|
GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
|
|
|
|
|
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
|
|
|
|
|
GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_DOWN;
|
|
|
|
|
GPIO_Init(PORT_SPI, &GPIO_InitStructure);
|
|
|
|
|
RCC_APB2PeriphClockCmd(RCC_APB2Periph_SPI1, ENABLE);
|
|
|
|
|
SPI_I2S_DeInit(SPI1);
|
|
|
|
|
SPI_InitStructure.SPI_Mode = SPI_Mode_Master;
|
|
|
|
|
SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex;
|
|
|
|
|
SPI_InitStructure.SPI_DataSize = SPI_DataSize_16b;
|
|
|
|
|
SPI_InitStructure.SPI_CPOL = SPI_CPOL_Low;
|
|
|
|
|
SPI_InitStructure.SPI_CPHA = SPI_CPHA_1Edge;
|
|
|
|
|
SPI_InitStructure.SPI_NSS = SPI_NSS_Soft;
|
|
|
|
|
SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_16;
|
|
|
|
|
SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB;
|
|
|
|
|
SPI_InitStructure.SPI_CRCPolynomial = 7;
|
|
|
|
|
SPI_Init(SPI1, &SPI_InitStructure);
|
|
|
|
|
SPI_Cmd(SPI1, ENABLE);
|
2024-06-29 15:22:55 +08:00
|
|
|
ltc1867_spi_select();
|
|
|
|
|
ltc1867_transfer(ltc1867_spi_data[LTC1867_CH_CNT - 1]);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ltc1867_ledir_on(void)
|
|
|
|
|
{
|
|
|
|
|
GPIO_SetBits(PORT_LEDIR, PIN_LEDIR);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ltc1867_ledir_off(void)
|
|
|
|
|
{
|
|
|
|
|
GPIO_ResetBits(PORT_LEDIR, PIN_LEDIR);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ltc1867_ledir_toggle(void)
|
|
|
|
|
{
|
|
|
|
|
GPIO_ToggleBits(PORT_LEDIR, PIN_LEDIR);
|
2024-05-01 21:43:05 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ltc1867_delay(void)
|
|
|
|
|
{
|
|
|
|
|
FlagStatus sts;
|
|
|
|
|
|
2024-06-29 15:22:55 +08:00
|
|
|
TIM_SetCounter(TIM3, 0);
|
|
|
|
|
TIM_ClearFlag(TIM3, TIM_FLAG_Update);
|
|
|
|
|
TIM_Cmd(TIM3, ENABLE);
|
2024-05-01 21:43:05 +08:00
|
|
|
while (1) {
|
2024-06-29 15:22:55 +08:00
|
|
|
sts = TIM_GetFlagStatus(TIM3, TIM_FLAG_Update);
|
2024-05-01 21:43:05 +08:00
|
|
|
if (sts != RESET) {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-06-29 15:22:55 +08:00
|
|
|
TIM_Cmd(TIM3, DISABLE);
|
2024-05-01 21:43:05 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ltc1867_conv(void)
|
|
|
|
|
{
|
|
|
|
|
GPIO_SetBits(PORT_LTC1867, PIN_CONV);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ltc1867_spi_select(void)
|
|
|
|
|
{
|
|
|
|
|
GPIO_ResetBits(PORT_LTC1867, PIN_SPI_CS);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint16_t ltc1867_transfer(uint16_t data_send)
|
|
|
|
|
{
|
|
|
|
|
FlagStatus flag;
|
|
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
|
flag = SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_TXE);
|
|
|
|
|
if (flag != RESET) {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
SPI_I2S_SendData(SPI1, data_send);
|
|
|
|
|
while (1) {
|
|
|
|
|
flag = SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_RXNE);
|
|
|
|
|
if (flag != RESET) {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return SPI_I2S_ReceiveData(SPI1);
|
|
|
|
|
}
|
|
|
|
|
|
2024-06-29 15:22:55 +08:00
|
|
|
void TIM2_IRQHandler(void)
|
2024-05-01 21:43:05 +08:00
|
|
|
{
|
2024-06-29 15:22:55 +08:00
|
|
|
uint16_t val[LTC1867_CH_CNT];
|
2024-05-01 21:43:05 +08:00
|
|
|
|
2024-06-29 15:22:55 +08:00
|
|
|
TIM_ClearITPendingBit(TIM2, TIM_IT_Update);
|
2024-05-01 21:43:05 +08:00
|
|
|
|
2024-06-29 15:22:55 +08:00
|
|
|
/* for control led IR */
|
|
|
|
|
ltc1867_samp_cnt++;
|
|
|
|
|
if (ltc1867_samp_cnt == (SAMPLES_PER_PERIOD / 2)) {
|
|
|
|
|
ltc1867_ledir_on();
|
|
|
|
|
} else if (ltc1867_samp_cnt >= SAMPLES_PER_PERIOD) {
|
|
|
|
|
ltc1867_ledir_off();
|
|
|
|
|
ltc1867_samp_cnt = 0;
|
|
|
|
|
}
|
2024-05-01 21:43:05 +08:00
|
|
|
|
2024-06-29 15:22:55 +08:00
|
|
|
/* sample multi channel adc data */
|
|
|
|
|
for (uint8_t i = 0; i < LTC1867_CH_CNT; i++) {
|
|
|
|
|
ltc1867_conv(); /* this point start convert */
|
|
|
|
|
ltc1867_delay(); /* delay 3.5us for convert time, reference for LTC1867 datasheet */
|
|
|
|
|
ltc1867_spi_select(); /* convert complete, make convert pin low for reading sample data by SPI */
|
|
|
|
|
val[i] = ltc1867_transfer(ltc1867_spi_data[i]); /* read convert data and send next channel information */
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
debug_write_data(DEBUG_CH_REF, (float)val[0]);
|
|
|
|
|
debug_write_data(DEBUG_CH_SIG, (float)val[1]);
|
|
|
|
|
// debug_write_data(DEBUG_CH_CNT, (float)ltc1867_samp_cnt * (65536 / SAMPLES_PER_PERIOD));
|
|
|
|
|
if (ltc1867_samp_cnt < SAMPLES_PER_PERIOD / 2) {
|
|
|
|
|
debug_write_data(DEBUG_CH_CNT, (float)0);
|
|
|
|
|
} else {
|
|
|
|
|
debug_write_data(DEBUG_CH_CNT, (float)65535);
|
|
|
|
|
}
|
2024-05-01 21:43:05 +08:00
|
|
|
debug_send_frame();
|
|
|
|
|
}
|