/* Megumin LED display firmware * Copyright (C) 2018 Sebastian Götte * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "global.h" #include "adc.h" #include "8seg_protocol.h" #include "transmit.h" volatile unsigned int sys_time = 0; volatile unsigned int sys_time_seconds = 0; uint16_t jitter_meas_avg_ns = 0; void TIM1_BRK_UP_TRG_COM_Handler() { TIM1->SR &= ~TIM_SR_UIF_Msk; } void set_drv_gpios(uint8_t val) { val = ~val; int a=!(val&1), b=!(val&2), c=!(val&4), d=!(val&8); GPIOA->BSRR = (((!a)<<3 | (!b)<<7 | (!c)<<6 | (!d)<<4)<<16) | ((a<<3) | (b<<7) | (c<<6) | (d<<4)); } uint8_t out_state = 0x0f; void set_outputs(uint8_t val[8]) { /* TODO implement BCM for digital brightness control */ int x = 0; for (int i=0; i<8; i++) if (val[i] > 127) x |= 1<BSRR = (1<<2) << (load ? 0 : 16); } void blank(void) { GPIOA->BRR = (1<<9) | (1<<10); set_drv_gpios(0); } bool has_sync = 0; void unblank_low(int bit) { if (backchannel_frame) { /* Set from protocol.c */ if (tx_next_bit() == 1) set_load(1); else /* 0; but also TX_IDLE */ set_load(0); } else if (has_sync) { if (bit) { //GPIOA->BSRR = (1<<10); set_drv_gpios(out_state & 0xf); } else { //GPIOA->BSRR = (1<<9); set_drv_gpios(out_state >> 4); } } } int sync_ctr = 0xffff; void TIM3_IRQHandler(void) { if (TIM3->SR & TIM_SR_CC2IF) { if (sync_ctr > 10) has_sync = 0; else sync_ctr += 1; EXTI->IMR = (1<<0); GPIOB->BSRR = (1<<1); GPIOA->BRR = (1<<9) | (1<<10); } else if (TIM3->SR & TIM_SR_CC3IF) { int bit = GPIOA->IDR & (1<<5); /* Sample current polarity */ unblank_low(!bit); } else { blank(); } TIM3->SR = 0; } void EXTI0_1_IRQHandler(void) { static uint32_t jitter_meas_sum = 0, jitter_meas_cnt = 0; EXTI->PR = (1<<0); /* Store old counter value for jitter measurement. Let it overflow to handle negative offsets. */ int16_t cnt = (int16_t)TIM3->CNT; /* Re-initialize the counter to align it with the signal edge */ TIM3->EGR |= TIM_EGR_UG; /* Don't handle overflow of _sum here since this value is only for monitoring anyway */ jitter_meas_sum += (cnt >= 0) ? cnt : -cnt; if (++jitter_meas_cnt == 1000) { /* One measurement roughly every 800ms */ jitter_meas_avg_ns = jitter_meas_sum; } EXTI->IMR = 0; GPIOB->BRR = (1<<1); has_sync = 1; sync_ctr = 0; } int main(void) { //RCC->CR |= RCC_CR_HSEON; //while (!(RCC->CR&RCC_CR_HSERDY)); RCC->CFGR &= ~RCC_CFGR_PLLMUL_Msk & ~RCC_CFGR_SW_Msk & ~RCC_CFGR_PPRE_Msk & ~RCC_CFGR_HPRE_Msk; RCC->CFGR |= ((12-2)< 48.0MHz */ RCC->CR |= RCC_CR_PLLON; while (!(RCC->CR&RCC_CR_PLLRDY)); RCC->CFGR |= (2<AHBENR |= RCC_AHBENR_DMAEN | RCC_AHBENR_GPIOAEN | RCC_AHBENR_GPIOBEN | RCC_AHBENR_FLITFEN; RCC->APB2ENR |= RCC_APB2ENR_SYSCFGEN | RCC_APB2ENR_ADCEN| RCC_APB2ENR_DBGMCUEN | RCC_APB2ENR_TIM1EN | RCC_APB2ENR_TIM1EN;; RCC->APB1ENR |= RCC_APB1ENR_TIM3EN; /* TIM3 foo */ TIM3->CCMR2 = (6<CCER = TIM_CCER_CC4E; /* Enable capture/compare unit 4 connected to ADC */ TIM3->CCER = TIM_CCER_CC3E; /* Enable capture/compare unit 3 for unblank interrupt */ TIM3->CCER = TIM_CCER_CC2E; TIM3->PSC = 48-1; /* 48MHz -> 1MHz */ TIM3->CCR2 = 800-1-1; TIM3->CCR3 = 100-1; /* CC3 is used for unblanking in the ISR, fires 30us after beginning of cycle. */ TIM3->CCR4 = 800-100-1; /* CC4 is ADC trigger, fire 30us before end of cycle. */ TIM3->ARR = 800-1; /* 1MHz -> 5kHz */ TIM3->DIER |= TIM_DIER_CC2IE | TIM_DIER_CC3IE | TIM_DIER_CC4IE | TIM_DIER_UIE; TIM3->CR1 |= TIM_CR1_CEN; NVIC_EnableIRQ(TIM3_IRQn); NVIC_SetPriority(TIM3_IRQn, 3<<5); GPIOB->MODER |= (1<OSPEEDR |= (2<IMR = (1<<0); /* PA0 Vmeas_A for sync */ EXTI->RTSR |= (1<<0); NVIC_EnableIRQ(EXTI0_1_IRQn); NVIC_SetPriority(EXTI0_1_IRQn, 4<<5); GPIOA->MODER |= (0<PUPDR |= (2<OSPEEDR |= (2< 10000) { cnt = 0; seg_c += 1; if (seg_c == 8) seg_c = 0; set_outputs_binary(1<