/* 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" volatile unsigned int sys_time = 0; volatile unsigned int sys_time_seconds = 0; void TIM1_BRK_UP_TRG_COM_Handler() { TIM1->SR &= ~TIM_SR_UIF_Msk; } enum packet_type { PKT_TYPE_RESERVED = 0, PKT_TYPE_SET_OUTPUTS_BINARY = 1, PKT_TYPE_SET_GLOBAL_BRIGHTNESS = 2, PKT_TYPE_SET_OUTPUTS = 3, PKT_TYPE_MAX }; struct { struct command_if_def cmd_if; int payload_len[PKT_TYPE_MAX]; } cmd_if = {{.packet_type_max=PKT_TYPE_MAX}, { [PKT_TYPE_RESERVED] = 0, [PKT_TYPE_SET_OUTPUTS_BINARY] = 1, [PKT_TYPE_SET_GLOBAL_BRIGHTNESS] = 1, [PKT_TYPE_SET_OUTPUTS] = 8 } }; void set_drv_gpios(uint8_t val) { int a=!!(val&1), b=!!(val&2), c=!!(val&4), d=!!(val&8); GPIOA->ODR &= ~(!a<<3 | !b<<7 | c<<6 | d<<4); GPIOA->ODR |= a<<3 | b<<7 | !c<<6 | !d<<4; } uint8_t out_state = 0x01; 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<DIER &= (~TIM_DIER_UIE) & (~TIM_DIER_CC4IE); TIM3->CCMR2 = (6<CCER = TIM_CCER_CC4E; /* Enable capture/compare unit 4 connected to ADC */ TIM3->CCR4 = 50; /* Trigger towards start of timer cycle */ TIM3->PSC = 48-1; TIM3->ARR = 400-1; TIM3->EGR |= TIM_EGR_UG; TIM3->CR1 = TIM_CR1_ARPE | TIM_CR1_OPM; TIM3->SR &= (~TIM_SR_UIF) & (~TIM_SR_CC4IF); TIM3->DIER |= TIM_DIER_UIE | TIM_DIER_CC4IE; TIM3->CR1 |= TIM_CR1_CEN; } void TIM3_IRQHandler(void) { if (TIM3->SR & TIM_SR_UIF) { blank(); } else { if (bit) set_drv_gpios(out_state & 0xf); else set_drv_gpios(out_state >> 4); } TIM3->SR = 0; } void handle_command(int command, uint8_t *args) { static int global_brightness = 0xff; switch (command) { case PKT_TYPE_SET_OUTPUTS_BINARY: set_outputs_binary(args[0], global_brightness); break; case PKT_TYPE_SET_GLOBAL_BRIGHTNESS: global_brightness = args[0]; break; case PKT_TYPE_SET_OUTPUTS: set_outputs(args); break; } } 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 |= ((6-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_FLITFEN; RCC->APB2ENR |= RCC_APB2ENR_SYSCFGEN | RCC_APB2ENR_ADCEN| RCC_APB2ENR_DBGMCUEN | RCC_APB2ENR_TIM1EN | RCC_APB2ENR_TIM1EN;; RCC->APB1ENR |= RCC_APB1ENR_TIM3EN; GPIOA->MODER |= (0<OSPEEDR |= (2<