// GPIO functions on stm32f4 // // Copyright (C) 2019 Kevin O'Connor // // This file may be distributed under the terms of the GNU GPLv3 license. #include // ffs #include "board/irq.h" // irq_save #include "command.h" // DECL_ENUMERATION_RANGE #include "gpio.h" // gpio_out_setup #include "internal.h" // gpio_peripheral #include "sched.h" // sched_shutdown DECL_ENUMERATION_RANGE("pin", "PA0", GPIO('A', 0), 32); DECL_ENUMERATION_RANGE("pin", "PB0", GPIO('B', 0), 32); DECL_ENUMERATION_RANGE("pin", "PC0", GPIO('C', 0), 32); static GPIO_TypeDef * const digital_regs[] = { GPIOA, GPIOB, GPIOC }; // Set the mode and extended function of a pin void gpio_peripheral(uint32_t gpio, uint32_t mode, int pullup) { GPIO_TypeDef *regs = digital_regs[GPIO2PORT(gpio)]; uint32_t mode_bits = mode & 0x0f, func = mode >> 4; uint32_t pup = pullup ? (pullup > 0 ? 1 : 2) : 0; uint32_t pos = gpio % 16, af_reg = pos / 8; uint32_t af_shift = (pos % 8) * 4, af_msk = 0x0f << af_shift; uint32_t m_shift = pos * 2, m_msk = 0x03 << m_shift; regs->AFR[af_reg] = (regs->AFR[af_reg] & ~af_msk) | (func << af_shift); regs->MODER = (regs->MODER & ~m_msk) | (mode_bits << m_shift); regs->PUPDR = (regs->PUPDR & ~m_msk) | (pup << m_shift); regs->OSPEEDR = (regs->OSPEEDR & ~m_msk) | (0x02 << m_shift); } // Convert a register and bit location back to an integer pin identifier static int regs_to_pin(GPIO_TypeDef *regs, uint32_t bit) { int i; for (i=0; i= ARRAY_SIZE(digital_regs)) goto fail; GPIO_TypeDef *regs = digital_regs[GPIO2PORT(pin)]; struct gpio_out g = { .regs=regs, .bit=GPIO2BIT(pin) }; gpio_out_reset(g, val); return g; fail: shutdown("Not an output pin"); } void gpio_out_reset(struct gpio_out g, uint32_t val) { GPIO_TypeDef *regs = g.regs; int pin = regs_to_pin(regs, g.bit); irqstatus_t flag = irq_save(); if (val) regs->BSRR = g.bit; else regs->BSRR = g.bit << 16; gpio_peripheral(pin, GPIO_OUTPUT, 0); irq_restore(flag); } void gpio_out_toggle_noirq(struct gpio_out g) { GPIO_TypeDef *regs = g.regs; regs->ODR ^= g.bit; } void gpio_out_toggle(struct gpio_out g) { irqstatus_t flag = irq_save(); gpio_out_toggle_noirq(g); irq_restore(flag); } void gpio_out_write(struct gpio_out g, uint32_t val) { GPIO_TypeDef *regs = g.regs; if (val) regs->BSRR = g.bit; else regs->BSRR = g.bit << 16; } struct gpio_in gpio_in_setup(uint32_t pin, int32_t pull_up) { if (GPIO2PORT(pin) >= ARRAY_SIZE(digital_regs)) goto fail; GPIO_TypeDef *regs = digital_regs[GPIO2PORT(pin)]; struct gpio_in g = { .regs=regs, .bit=GPIO2BIT(pin) }; gpio_in_reset(g, pull_up); return g; fail: shutdown("Not a valid input pin"); } void gpio_in_reset(struct gpio_in g, int32_t pull_up) { GPIO_TypeDef *regs = g.regs; int pin = regs_to_pin(regs, g.bit); irqstatus_t flag = irq_save(); gpio_peripheral(pin, GPIO_INPUT, pull_up); irq_restore(flag); } uint8_t gpio_in_read(struct gpio_in g) { GPIO_TypeDef *regs = g.regs; return !!(regs->IDR & g.bit); }