mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-07 16:38:30 -04:00
242 lines
7.8 KiB
C++
242 lines
7.8 KiB
C++
#include <AP_HAL/AP_HAL.h>
|
|
|
|
#include <fcntl.h>
|
|
#include <poll.h>
|
|
#include <sys/mman.h>
|
|
#include <sys/stat.h>
|
|
#include <unistd.h>
|
|
|
|
#include "GPIO.h"
|
|
#include "Util_RPI.h"
|
|
#include "GPIO_RPI_BCM.h"
|
|
|
|
#define GPIO_RPI_MAX_NUMBER_PINS 32
|
|
|
|
using namespace Linux;
|
|
|
|
extern const AP_HAL::HAL& hal;
|
|
|
|
// Range based in the first memory address of the first register and the last memory addres
|
|
// for the GPIO section (0x7E20'00B4 - 0x7E20'0000).
|
|
const uint8_t GPIO_RPI_BCM::_gpio_registers_memory_range = 0xB4;
|
|
const char* GPIO_RPI_BCM::_system_memory_device_path = "/dev/mem";
|
|
|
|
GPIO_RPI_BCM::GPIO_RPI_BCM()
|
|
{
|
|
}
|
|
|
|
void GPIO_RPI_BCM::set_gpio_mode_alt(int pin, int alternative)
|
|
{
|
|
// Each register can contain 10 pins
|
|
const uint8_t pins_per_register = 10;
|
|
// Calculates the position of the 3 bit mask in the 32 bits register
|
|
const uint8_t tree_bits_position_in_register = (pin%pins_per_register)*3;
|
|
/** Creates a mask to enable the alternative function based in the following logic:
|
|
*
|
|
* | Alternative Function | 3 bits value |
|
|
* |:--------------------:|:------------:|
|
|
* | Function 0 | 0b100 |
|
|
* | Function 1 | 0b101 |
|
|
* | Function 2 | 0b110 |
|
|
* | Function 3 | 0b111 |
|
|
* | Function 4 | 0b011 |
|
|
* | Function 5 | 0b010 |
|
|
*/
|
|
const uint8_t alternative_value =
|
|
(alternative < 4 ? (alternative + 4) : (alternative == 4 ? 3 : 2));
|
|
// 0b00'000'000'000'000'000'000'ALT'000'000'000 enables alternative for the 4th pin
|
|
const uint32_t mask_with_alt = static_cast<uint32_t>(alternative_value) << tree_bits_position_in_register;
|
|
const uint32_t mask = 0b111 << tree_bits_position_in_register;
|
|
// Clear all bits in our position and apply our mask with alt values
|
|
uint32_t register_value = _gpio[pin / pins_per_register];
|
|
register_value &= ~mask;
|
|
_gpio[pin / pins_per_register] = register_value | mask_with_alt;
|
|
}
|
|
|
|
void GPIO_RPI_BCM::set_gpio_mode_in(int pin)
|
|
{
|
|
// Each register can contain 10 pins
|
|
const uint8_t pins_per_register = 10;
|
|
// Calculates the position of the 3 bit mask in the 32 bits register
|
|
const uint8_t tree_bits_position_in_register = (pin%pins_per_register)*3;
|
|
// Create a mask that only removes the bits in this specific GPIO pin, E.g:
|
|
// 0b11'111'111'111'111'111'111'000'111'111'111 for the 4th pin
|
|
const uint32_t mask = ~(0b111<<tree_bits_position_in_register);
|
|
// Apply mask
|
|
_gpio[pin / pins_per_register] &= mask;
|
|
}
|
|
|
|
void GPIO_RPI_BCM::set_gpio_mode_out(int pin)
|
|
{
|
|
// Each register can contain 10 pins
|
|
const uint8_t pins_per_register = 10;
|
|
// Calculates the position of the 3 bit mask in the 32 bits register
|
|
const uint8_t tree_bits_position_in_register = (pin%pins_per_register)*3;
|
|
// Create a mask to enable the bit that sets output functionality
|
|
// 0b00'000'000'000'000'000'000'001'000'000'000 enables output for the 4th pin
|
|
const uint32_t mask_with_bit = 0b001 << tree_bits_position_in_register;
|
|
const uint32_t mask = 0b111 << tree_bits_position_in_register;
|
|
// Clear all bits in our position and apply our mask with alt values
|
|
uint32_t register_value = _gpio[pin / pins_per_register];
|
|
register_value &= ~mask;
|
|
_gpio[pin / pins_per_register] = register_value | mask_with_bit;
|
|
}
|
|
|
|
void GPIO_RPI_BCM::set_gpio_high(int pin)
|
|
{
|
|
// Calculate index of the array for the register GPSET0 (0x7E20'001C)
|
|
constexpr uint32_t gpset0_memory_offset_value = 0x1c;
|
|
constexpr uint32_t gpset0_index_value = gpset0_memory_offset_value / sizeof(*_gpio);
|
|
_gpio[gpset0_index_value] = 1 << pin;
|
|
}
|
|
|
|
void GPIO_RPI_BCM::set_gpio_low(int pin)
|
|
{
|
|
// Calculate index of the array for the register GPCLR0 (0x7E20'0028)
|
|
constexpr uint32_t gpclr0_memory_offset_value = 0x28;
|
|
constexpr uint32_t gpclr0_index_value = gpclr0_memory_offset_value / sizeof(*_gpio);
|
|
_gpio[gpclr0_index_value] = 1 << pin;
|
|
}
|
|
|
|
bool GPIO_RPI_BCM::get_gpio_logic_state(int pin)
|
|
{
|
|
// Calculate index of the array for the register GPLEV0 (0x7E20'0034)
|
|
constexpr uint32_t gplev0_memory_offset_value = 0x34;
|
|
constexpr uint32_t gplev0_index_value = gplev0_memory_offset_value / sizeof(*_gpio);
|
|
return _gpio[gplev0_index_value] & (1 << pin);
|
|
}
|
|
|
|
uint32_t GPIO_RPI_BCM::get_address(GPIO_RPI_BCM::Address address, GPIO_RPI_BCM::PeripheralOffset offset) const
|
|
{
|
|
return static_cast<uint32_t>(address) + static_cast<uint32_t>(offset);
|
|
}
|
|
|
|
volatile uint32_t* GPIO_RPI_BCM::get_memory_pointer(uint32_t address, uint32_t range) const
|
|
{
|
|
auto pointer = mmap(
|
|
nullptr, // Any adddress in our space will do
|
|
range, // Map length
|
|
PROT_READ|PROT_WRITE|PROT_EXEC, // Enable reading & writing to mapped memory
|
|
MAP_SHARED|MAP_LOCKED, // Shared with other processes
|
|
_system_memory_device, // File to map
|
|
address // Offset to GPIO peripheral
|
|
);
|
|
|
|
if (pointer == MAP_FAILED) {
|
|
return nullptr;
|
|
}
|
|
|
|
return static_cast<volatile uint32_t*>(pointer);
|
|
}
|
|
|
|
bool GPIO_RPI_BCM::openMemoryDevice()
|
|
{
|
|
_system_memory_device = open(_system_memory_device_path, O_RDWR|O_SYNC|O_CLOEXEC);
|
|
if (_system_memory_device < 0) {
|
|
AP_HAL::panic("Can't open %s", GPIO_RPI_BCM::_system_memory_device_path);
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void GPIO_RPI_BCM::closeMemoryDevice()
|
|
{
|
|
close(_system_memory_device);
|
|
// Invalidate device variable
|
|
_system_memory_device = -1;
|
|
}
|
|
|
|
void GPIO_RPI_BCM::init()
|
|
{
|
|
const LINUX_BOARD_TYPE rpi_version = UtilRPI::from(hal.util)->detect_linux_board_type();
|
|
|
|
GPIO_RPI_BCM::Address peripheral_base;
|
|
if(rpi_version == LINUX_BOARD_TYPE::RPI_ZERO_1) {
|
|
peripheral_base = Address::BCM2708_PERIPHERAL_BASE;
|
|
} else if (rpi_version == LINUX_BOARD_TYPE::RPI_2_3_ZERO2) {
|
|
peripheral_base = Address::BCM2709_PERIPHERAL_BASE;
|
|
} else if (rpi_version == LINUX_BOARD_TYPE::RPI_4) {
|
|
peripheral_base = Address::BCM2711_PERIPHERAL_BASE;
|
|
} else {
|
|
AP_HAL::panic("Unknown rpi_version, cannot locate peripheral base address");
|
|
return;
|
|
}
|
|
|
|
if (!openMemoryDevice()) {
|
|
AP_HAL::panic("Failed to initialize memory device.");
|
|
return;
|
|
}
|
|
|
|
const uint32_t gpio_address = get_address(peripheral_base, PeripheralOffset::GPIO);
|
|
|
|
_gpio = get_memory_pointer(gpio_address, _gpio_registers_memory_range);
|
|
if (!_gpio) {
|
|
AP_HAL::panic("Failed to get GPIO memory map.");
|
|
}
|
|
|
|
// No need to keep mem_fd open after mmap
|
|
closeMemoryDevice();
|
|
}
|
|
|
|
void GPIO_RPI_BCM::pinMode(uint8_t pin, uint8_t output)
|
|
{
|
|
if (output == HAL_GPIO_INPUT) {
|
|
set_gpio_mode_in(pin);
|
|
} else {
|
|
set_gpio_mode_in(pin);
|
|
set_gpio_mode_out(pin);
|
|
}
|
|
}
|
|
|
|
void GPIO_RPI_BCM::pinMode(uint8_t pin, uint8_t output, uint8_t alt)
|
|
{
|
|
if (output == HAL_GPIO_INPUT) {
|
|
set_gpio_mode_in(pin);
|
|
} else if (output == HAL_GPIO_ALT) {
|
|
set_gpio_mode_in(pin);
|
|
set_gpio_mode_alt(pin, alt);
|
|
} else {
|
|
set_gpio_mode_in(pin);
|
|
set_gpio_mode_out(pin);
|
|
}
|
|
}
|
|
|
|
uint8_t GPIO_RPI_BCM::read(uint8_t pin)
|
|
{
|
|
if (pin >= GPIO_RPI_MAX_NUMBER_PINS) {
|
|
return 0;
|
|
}
|
|
return static_cast<uint8_t>(get_gpio_logic_state(pin));
|
|
}
|
|
|
|
void GPIO_RPI_BCM::write(uint8_t pin, uint8_t value)
|
|
{
|
|
if (value != 0) {
|
|
set_gpio_high(pin);
|
|
} else {
|
|
set_gpio_low(pin);
|
|
}
|
|
}
|
|
|
|
void GPIO_RPI_BCM::toggle(uint8_t pin)
|
|
{
|
|
if (pin >= GPIO_RPI_MAX_NUMBER_PINS) {
|
|
return ;
|
|
}
|
|
uint32_t flag = (1 << pin);
|
|
_gpio_output_port_status ^= flag;
|
|
write(pin, (_gpio_output_port_status & flag) >> pin);
|
|
}
|
|
|
|
/* Alternative interface: */
|
|
AP_HAL::DigitalSource* GPIO_RPI_BCM::channel(uint16_t n)
|
|
{
|
|
return NEW_NOTHROW DigitalSource(n);
|
|
}
|
|
|
|
bool GPIO_RPI_BCM::usb_connected(void)
|
|
{
|
|
return false;
|
|
}
|