mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-14 20:58:30 -04:00
184 lines
4.1 KiB
C++
184 lines
4.1 KiB
C++
|
/*
|
||
|
(c) 2017 night_ghost@ykoctpa.ru
|
||
|
|
||
|
*/
|
||
|
|
||
|
#pragma GCC optimize ("O2")
|
||
|
|
||
|
#include "gpio_hal.h"
|
||
|
#include <boards.h>
|
||
|
|
||
|
#include <exti.h>
|
||
|
|
||
|
#include "GPIO.h"
|
||
|
#include "Scheduler.h"
|
||
|
#include "RCOutput.h"
|
||
|
|
||
|
|
||
|
|
||
|
using namespace F4Light;
|
||
|
|
||
|
void GPIO::_pinMode(uint8_t pin, uint8_t mode)
|
||
|
{
|
||
|
gpio_pin_mode outputMode;
|
||
|
bool pwm = false;
|
||
|
|
||
|
switch(mode) { // modes defined to be compatible so no transcode required
|
||
|
case OUTPUT:
|
||
|
case OUTPUT_OPEN_DRAIN:
|
||
|
case INPUT:
|
||
|
// case INPUT_FLOATING: synonim and cause doubled 'case'
|
||
|
case INPUT_ANALOG:
|
||
|
case INPUT_PULLUP:
|
||
|
case INPUT_PULLDOWN:
|
||
|
outputMode = (gpio_pin_mode)mode;
|
||
|
break;
|
||
|
|
||
|
case PWM:
|
||
|
outputMode = GPIO_AF_OUTPUT_PP;
|
||
|
pwm = true;
|
||
|
break;
|
||
|
|
||
|
case PWM_OPEN_DRAIN:
|
||
|
outputMode = GPIO_AF_OUTPUT_OD;
|
||
|
pwm = true;
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
assert_param(0);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
|
||
|
const stm32_pin_info &p = PIN_MAP[pin];
|
||
|
|
||
|
const gpio_dev* dev = p.gpio_device;
|
||
|
uint8_t bit = p.gpio_bit;
|
||
|
const timer_dev * timer = p.timer_device;
|
||
|
|
||
|
gpio_set_mode(dev, bit, outputMode);
|
||
|
|
||
|
if (pwm && timer != NULL) {
|
||
|
gpio_set_speed(dev, bit, GPIO_speed_25MHz); // cleanflight sets 2MHz
|
||
|
gpio_set_af_mode(dev, bit, timer->af);
|
||
|
timer_set_mode(timer, p.timer_channel, TIMER_PWM); // init in setupTimers()
|
||
|
} else {
|
||
|
gpio_set_af_mode(dev, bit, 0); // reset
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
void GPIO::pinMode(uint8_t pin, uint8_t output){
|
||
|
|
||
|
if ((pin >= BOARD_NR_GPIO_PINS)) return;
|
||
|
|
||
|
_pinMode(pin, output);
|
||
|
}
|
||
|
|
||
|
|
||
|
uint8_t GPIO::read(uint8_t pin) {
|
||
|
if (pin >= BOARD_NR_GPIO_PINS) return 0;
|
||
|
|
||
|
return _read(pin);
|
||
|
}
|
||
|
|
||
|
|
||
|
void GPIO::write(uint8_t pin, uint8_t value) {
|
||
|
if ((pin >= BOARD_NR_GPIO_PINS)) return;
|
||
|
|
||
|
#ifdef BUZZER_PWM_HZ // passive buzzer
|
||
|
|
||
|
// AP_Notify Buzzer.cpp don't supports passive buzzer so we need a small hack
|
||
|
if(pin == BOARD_BUZZER_PIN){
|
||
|
if(value == HAL_BUZZER_ON){
|
||
|
const stm32_pin_info &p = PIN_MAP[pin];
|
||
|
const timer_dev *dev = p.timer_device;
|
||
|
if(dev->state->freq==0) {
|
||
|
configTimeBase(dev, 0, BUZZER_PWM_HZ * 10); // it should be personal timer
|
||
|
}
|
||
|
_pinMode(pin, PWM);
|
||
|
uint32_t n = RCOutput::_timer_period(BUZZER_PWM_HZ, dev);
|
||
|
timer_set_reload(dev, n);
|
||
|
timer_set_compare(dev, p.timer_channel, n/2);
|
||
|
return;
|
||
|
} else {
|
||
|
_pinMode(pin, OUTPUT); // to disable, just change mode
|
||
|
}
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
_write(pin, value);
|
||
|
}
|
||
|
|
||
|
|
||
|
void GPIO::toggle(uint8_t pin)
|
||
|
{
|
||
|
if ((pin >= BOARD_NR_GPIO_PINS)) return;
|
||
|
|
||
|
_toggle(pin);
|
||
|
}
|
||
|
|
||
|
|
||
|
/* Interrupt interface: */
|
||
|
bool GPIO::_attach_interrupt(uint8_t pin, Handler p, uint8_t mode, uint8_t priority)
|
||
|
{
|
||
|
if ( (pin >= BOARD_NR_GPIO_PINS) || !p) return false;
|
||
|
|
||
|
const stm32_pin_info &pp = PIN_MAP[pin];
|
||
|
|
||
|
exti_attach_interrupt_pri((afio_exti_num)(pp.gpio_bit),
|
||
|
gpio_exti_port(pp.gpio_device),
|
||
|
p, exti_out_mode((ExtIntTriggerMode)mode),
|
||
|
priority);
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
void GPIO::detach_interrupt(uint8_t pin)
|
||
|
{
|
||
|
if ( pin >= BOARD_NR_GPIO_PINS) return;
|
||
|
|
||
|
exti_detach_interrupt((afio_exti_num)(PIN_MAP[pin].gpio_bit));
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
/* Alternative interface: */
|
||
|
AP_HAL::DigitalSource* GPIO::channel(uint16_t pin) {
|
||
|
|
||
|
if ((pin >= BOARD_NR_GPIO_PINS)) return NULL;
|
||
|
|
||
|
return get_channel(pin);
|
||
|
}
|
||
|
|
||
|
|
||
|
void DigitalSource::mode(uint8_t md)
|
||
|
{
|
||
|
gpio_pin_mode outputMode;
|
||
|
|
||
|
switch(md) {
|
||
|
case OUTPUT:
|
||
|
case OUTPUT_OPEN_DRAIN:
|
||
|
case INPUT:
|
||
|
// case INPUT_FLOATING:
|
||
|
case INPUT_ANALOG:
|
||
|
case INPUT_PULLUP:
|
||
|
case INPUT_PULLDOWN:
|
||
|
outputMode = (gpio_pin_mode)md;
|
||
|
break;
|
||
|
|
||
|
// no PWM via this interface!
|
||
|
default:
|
||
|
assert_param(0);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
gpio_set_mode( _device, _bit, outputMode);
|
||
|
gpio_set_speed(_device, _bit, GPIO_speed_100MHz); // to use as CS
|
||
|
}
|
||
|
|
||
|
void digitalWrite(uint8_t pin, uint8_t value) { F4Light::GPIO::_write(pin, value); }
|
||
|
uint8_t digitalRead(uint8_t pin) { return F4Light::GPIO::_read(pin); }
|
||
|
|
||
|
void digitalToggle(uint8_t pin) { return F4Light::GPIO::_toggle(pin); }
|