@@ 9,12 9,17 @@
#include "exti.h"
#include "registers.h"
#include "pin.h"
+#include "fmc.h"
#define LED1_GPIO GPIOI
#define LED1_PIN 12
/* #define LED2_PIN 13 */
void hard_fault_handler() {
+ volatile uint32_t cfsr = SCB->CFSR;
+ volatile uint32_t hfsr = SCB->HFSR;
+ volatile uint32_t bfar = SCB->BFAR;
+
while(1) {}
}
@@ 43,9 48,6 @@ void led_input()
void led_gpio_en()
{
RCC->AHB4ENR |= (1 << RCC_AHB4ENR_GPIOIEN_Pos);
- volatile uint32_t dummy;
- dummy = RCC->AHB4ENR;
- dummy = RCC->AHB4ENR;
}
void exti15_10_handler(void)
@@ 58,48 60,215 @@ void exti15_10_handler(void)
void main()
{
+ clocks_wait_ready(CLOCK_HSI);
+
+ // Clock gating
+ RCC->APB4ENR |= RCC_APB4ENR_SYSCFGEN;
+ RCC->AHB4ENR |= RCC_AHB4ENR_GPIOAEN | RCC_AHB4ENR_GPIOBEN |
+ RCC_AHB4ENR_GPIOCEN | RCC_AHB4ENR_GPIODEN |
+ RCC_AHB4ENR_GPIOEEN | RCC_AHB4ENR_GPIOFEN |
+ RCC_AHB4ENR_GPIOGEN | RCC_AHB4ENR_GPIOHEN |
+ RCC_AHB4ENR_GPIOIEN | RCC_AHB4ENR_GPIOJEN;
+
systick_configure();
// Clocks section
// Enable hsi48 for usb
- RCC->CR |= RCC_CR_HSI48ON;
- while ((RCC->CR & RCC_CR_HSI48RDY) == 0);
-
- // TODO: pll, system clock switch... too complicated it seems.
- // Getting hard faults, apparently because of too low voltage
- // Has to be enabled before vco can be changed
- /* PWR->CR3 |= PWR_CR3_LDOEN; */
- /* PWR->CR3 &= ~PWR_CR3_LDOEN; */
- /* reg_write_bits_pos(&PWR->D3CR, 2, PWR_D3CR_VOS_Pos, 3); */
-
- /* /\* SYSCFG->PWRCR |= SYSCFG_PWRCR_ODEN; *\/ */
-
- /* while((PWR->CSR1 & PWR_CSR1_ACTVOSRDY) == 0); */
- /* /\* while((PWR->D3CR & PWR_D3CR_VOSRDY) == 0); *\/ */
-
- /* reg_write_bits(&FLASH->ACR, FLASH_ACR_LATENCY_3WS, FLASH_ACR_LATENCY_Msk); */
-
- /* // HSI is 64 MHz, not divided */
- /* // Diving by 32 to put to PLL -> 2 MHz */
- /* // DIVN = 360 -> F_VCO = 720 MHz */
- /* // DIVP = 2 -> pll1_p is 360 MHz */
- /* // DIVQ = 8 -> pll1_q is 90 MHz */
- /* // DIVR = 8 -> pll1_r is 90 MHz */
- /* clocks_pll_configure(CLOCK_PLL1, 32, PLL_SOURCE_HSI, */
- /* 360, 2, 8, 8); */
- /* clocks_pll_enable(CLOCK_PLL1); */
- /* clocks_pll_wait_ready(CLOCK_PLL1, 300); */
- /* clocks_system_clock_source(CLOCK_SOURCE_PLL_1_P_CK, */
- /* 1, 1, 2, 300); */
+ clocks_enable(CLOCK_HSI48);
+ clocks_wait_ready(CLOCK_HSI48);
- // Clock gating
- RCC->APB4ENR |= RCC_APB4ENR_SYSCFGEN;
- volatile uint32_t dummy;
- dummy = RCC->APB4ENR;
- dummy = RCC->APB4ENR;
- RCC->AHB4ENR |= RCC_AHB4ENR_GPIOAEN | RCC_AHB4ENR_GPIOBEN | RCC_AHB4ENR_GPIOCEN | RCC_AHB4ENR_GPIOHEN | RCC_AHB4ENR_GPIOIEN | RCC_AHB4ENR_GPIOJEN;
- dummy = RCC->AHB4ENR;
- dummy = RCC->AHB4ENR;
+ clocks_enable(CLOCK_HSE);
+ clocks_wait_ready(CLOCK_HSE);
+
+ // HSE 25 MHz -> divide by 25 -> 1 MHz
+ // HSE / 25 * 400 = FVCO = 400 MHz
+ // PLL2 R = 200 MHz
+ reg_set_bits(&RCC->PLLCFGR, RCC_PLLCFGR_PLL3VCOSEL_Msk);
+ clocks_pll_configure(CLOCK_PLL2, 25, PLL_SOURCE_HSE, 400, 2, 2, 2);
+ clocks_pll_enable(CLOCK_PLL2);
+ clocks_pll_wait_ready(CLOCK_PLL2, 300);
+
+ fmc_t fmc;
+ // select pll 2 r as fmc clock
+ reg_write_bits(&RCC->D1CCIPR, 2 << (RCC_D1CCIPR_FMCSEL_Pos), RCC_D1CCIPR_FMCSEL_Msk);
+ RCC->AHB3ENR |= RCC_AHB3ENR_FMCEN;
+ fmc_init(&fmc);
+
+ #define BURST_LENGTH_1 0x0000
+ #define BURST_TYPE_SEQUENTIAL 0x0000
+ #define CAS_LATENCY_3 0x0030
+ #define OPERATING_MODE_STANDARD 0x0000
+ #define WRITEBURST_MODE_SINGLE 0x0200
+ uint16_t sdram_mode_register = BURST_LENGTH_1 | BURST_TYPE_SEQUENTIAL |
+ CAS_LATENCY_3 | OPERATING_MODE_STANDARD |
+ WRITEBURST_MODE_SINGLE;
+
+ fmc_sdram_timing_t sdram_timing = {
+ .startup_delay_us = 100, // 100 µs
+ .max_sd_clock_hz = 100000000, // 100 MHz
+ .refresh_period_ns = 15625, // 64ms / (4096 rows) = 15625ns
+ .mode_register_to_active = 2, // tMRD = 2 cycles
+ .exit_self_refresh = 7, // tXSR = 70ns
+ .active_to_precharge = 4, // tRAS = 42ns
+ .row_cycle = 7, // tRC = 70ns
+ .row_precharge = 2, // tRP = 18ns
+ .row_to_column = 2, // tRCD = 18ns
+ };
+ fmc_sdram_configuration_t sdram_config = {
+ .column_bits = 9,
+ .row_bits = 12,
+ .memory_data_width = 32, // 32-bit
+ .internal_banks = 4, // 4 internal banks
+ .cas_latency = 3, // CAS latency = 3
+ .write_protection = false,
+ .read_burst = true,
+ .read_pipe_delay_cycles = 0,
+ };
+
+ {
+ pin_t pin_addr[12];
+ pin_t pin_data[32];
+ pin_t pin_ba[2];
+ pin_t pin_nbl[4];
+ pin_t pin_sdcke1;
+ pin_t pin_sdclk;
+ pin_t pin_sdncas;
+ pin_t pin_sdne1;
+ pin_t pin_sdras;
+ pin_t pin_sdnwe;
+
+ // A0-A11
+ pin_init(&pin_addr[0], GPIOF, 0);
+ pin_init(&pin_addr[1], GPIOF, 1);
+ pin_init(&pin_addr[2], GPIOF, 2);
+ pin_init(&pin_addr[3], GPIOF, 3);
+ pin_init(&pin_addr[4], GPIOF, 4);
+ pin_init(&pin_addr[5], GPIOF, 5);
+ pin_init(&pin_addr[6], GPIOF, 12);
+ pin_init(&pin_addr[7], GPIOF, 13);
+ pin_init(&pin_addr[8], GPIOF, 14);
+ pin_init(&pin_addr[9], GPIOF, 15);
+ pin_init(&pin_addr[10], GPIOG, 0);
+ pin_init(&pin_addr[11], GPIOG, 1);
+ // BA0-BA1
+ pin_init(&pin_ba[0], GPIOG, 4);
+ pin_init(&pin_ba[1], GPIOG, 5);
+ // D0-D31
+ pin_init(&pin_data[0], GPIOD, 14);
+ pin_init(&pin_data[1], GPIOD, 15);
+ pin_init(&pin_data[2], GPIOD, 0);
+ pin_init(&pin_data[3], GPIOD, 1);
+ pin_init(&pin_data[4], GPIOE, 7);
+ pin_init(&pin_data[5], GPIOE, 8);
+ pin_init(&pin_data[6], GPIOE, 9);
+ pin_init(&pin_data[7], GPIOE, 10);
+ pin_init(&pin_data[8], GPIOE, 11);
+ pin_init(&pin_data[9], GPIOE, 12);
+ pin_init(&pin_data[10], GPIOE, 13);
+ pin_init(&pin_data[11], GPIOE, 14);
+ pin_init(&pin_data[12], GPIOE, 15);
+ pin_init(&pin_data[13], GPIOD, 8);
+ pin_init(&pin_data[14], GPIOD, 9);
+ pin_init(&pin_data[15], GPIOD, 10);
+ pin_init(&pin_data[16], GPIOH, 8);
+ pin_init(&pin_data[17], GPIOH, 9);
+ pin_init(&pin_data[18], GPIOH, 10);
+ pin_init(&pin_data[19], GPIOH, 11);
+ pin_init(&pin_data[20], GPIOH, 12);
+ pin_init(&pin_data[21], GPIOH, 13);
+ pin_init(&pin_data[22], GPIOH, 14);
+ pin_init(&pin_data[23], GPIOH, 15);
+ pin_init(&pin_data[24], GPIOI, 0);
+ pin_init(&pin_data[25], GPIOI, 1);
+ pin_init(&pin_data[26], GPIOI, 2);
+ pin_init(&pin_data[27], GPIOI, 3);
+ pin_init(&pin_data[28], GPIOI, 6);
+ pin_init(&pin_data[29], GPIOI, 7);
+ pin_init(&pin_data[30], GPIOI, 9);
+ pin_init(&pin_data[31], GPIOI, 10);
+
+ pin_init(&pin_nbl[0], GPIOE, 0);
+ pin_init(&pin_nbl[1], GPIOE, 1);
+ pin_init(&pin_nbl[2], GPIOI, 4);
+ pin_init(&pin_nbl[3], GPIOI, 5);
+
+ pin_init(&pin_sdcke1, GPIOH, 7);
+ pin_init(&pin_sdclk, GPIOG, 8);
+ pin_init(&pin_sdncas, GPIOG, 15);
+ pin_init(&pin_sdne1, GPIOH, 6);
+ pin_init(&pin_sdras, GPIOF, 11);
+ pin_init(&pin_sdnwe, GPIOH, 5);
+
+ // A0-A11
+ pin_into_alternate_highspeed(&pin_addr[0], 12);
+ pin_into_alternate_highspeed(&pin_addr[1], 12);
+ pin_into_alternate_highspeed(&pin_addr[2], 12);
+ pin_into_alternate_highspeed(&pin_addr[3], 12);
+ pin_into_alternate_highspeed(&pin_addr[4], 12);
+ pin_into_alternate_highspeed(&pin_addr[5], 12);
+ pin_into_alternate_highspeed(&pin_addr[6], 12);
+ pin_into_alternate_highspeed(&pin_addr[7], 12);
+ pin_into_alternate_highspeed(&pin_addr[8], 12);
+ pin_into_alternate_highspeed(&pin_addr[9], 12);
+ pin_into_alternate_highspeed(&pin_addr[10], 12);
+ pin_into_alternate_highspeed(&pin_addr[11], 12);
+ // BA0-BA1
+ pin_into_alternate_highspeed(&pin_ba[0], 12);
+ pin_into_alternate_highspeed(&pin_ba[1], 12);
+ // D0-D31
+ pin_into_alternate_highspeed(&pin_data[0], 12);
+ pin_into_alternate_highspeed(&pin_data[1], 12);
+ pin_into_alternate_highspeed(&pin_data[2], 12);
+ pin_into_alternate_highspeed(&pin_data[3], 12);
+ pin_into_alternate_highspeed(&pin_data[4], 12);
+ pin_into_alternate_highspeed(&pin_data[5], 12);
+ pin_into_alternate_highspeed(&pin_data[6], 12);
+ pin_into_alternate_highspeed(&pin_data[7], 12);
+ pin_into_alternate_highspeed(&pin_data[8], 12);
+ pin_into_alternate_highspeed(&pin_data[9], 12);
+ pin_into_alternate_highspeed(&pin_data[10], 12);
+ pin_into_alternate_highspeed(&pin_data[11], 12);
+ pin_into_alternate_highspeed(&pin_data[12], 12);
+ pin_into_alternate_highspeed(&pin_data[13], 12);
+ pin_into_alternate_highspeed(&pin_data[14], 12);
+ pin_into_alternate_highspeed(&pin_data[15], 12);
+ pin_into_alternate_highspeed(&pin_data[16], 12);
+ pin_into_alternate_highspeed(&pin_data[17], 12);
+ pin_into_alternate_highspeed(&pin_data[18], 12);
+ pin_into_alternate_highspeed(&pin_data[19], 12);
+ pin_into_alternate_highspeed(&pin_data[20], 12);
+ pin_into_alternate_highspeed(&pin_data[21], 12);
+ pin_into_alternate_highspeed(&pin_data[22], 12);
+ pin_into_alternate_highspeed(&pin_data[23], 12);
+ pin_into_alternate_highspeed(&pin_data[24], 12);
+ pin_into_alternate_highspeed(&pin_data[25], 12);
+ pin_into_alternate_highspeed(&pin_data[26], 12);
+ pin_into_alternate_highspeed(&pin_data[27], 12);
+ pin_into_alternate_highspeed(&pin_data[28], 12);
+ pin_into_alternate_highspeed(&pin_data[29], 12);
+ pin_into_alternate_highspeed(&pin_data[30], 12);
+ pin_into_alternate_highspeed(&pin_data[31], 12);
+
+ pin_into_alternate_highspeed(&pin_nbl[0], 12);
+ pin_into_alternate_highspeed(&pin_nbl[1], 12);
+ pin_into_alternate_highspeed(&pin_nbl[2], 12);
+ pin_into_alternate_highspeed(&pin_nbl[3], 12);
+
+ pin_into_alternate_highspeed(&pin_sdcke1, 12);
+ pin_into_alternate_highspeed(&pin_sdclk, 12);
+ pin_into_alternate_highspeed(&pin_sdncas, 12);
+ pin_into_alternate_highspeed(&pin_sdne1, 12);
+ pin_into_alternate_highspeed(&pin_sdras, 12);
+ pin_into_alternate_highspeed(&pin_sdnwe, 12);
+
+ fmc_sdram_configure(&fmc, SDRAM_BANK2, sdram_config, sdram_timing, sdram_mode_register);
+ }
+
+ uint32_t* sdram_data = (uint32_t*)fmc_sdram_allocate(&fmc, SDRAM_BANK2, 1024);
+
+ for (uint32_t i = 0; i < 1024/4; i++) {
+ *(sdram_data + i) = i;
+ }
// Pins init
pin_t led1;
@@ 118,8 287,6 @@ void main()
exti_external_interrupt(&exti_wkup, EXTI_GPIOC);
exti_rising_interrupt(&exti_wkup);
exti_enable_interrupt(&exti_wkup);
- /* NVIC_SetPriority(EXTI15_10_IRQn, 2); */
- /* NVIC_EnableIRQ(EXTI15_10_IRQn); */
__enable_irq();
@@ 154,10 321,8 @@ void main()
pin_into_alternate_highspeed(&ulpi_d6, 10);
pin_into_alternate_highspeed(&ulpi_d7, 10);
- // TODO: ?
pin_into_input_highspeed(&otg_hs_overcurrent);
- // TODO: why can't I send string descriptors?
void *usb_dev = usb_device_init(USB_OTG_HS1, &USB_CLASS_CDC_ACM,
0x1234, 0x1111, u"Frantisek Bohacek",
u"Display", 1, NULL);