Atmosphere/sept/sept-primary/src/gpio.c

87 lines
2.9 KiB
C
Raw Normal View History

2019-02-20 12:52:11 +00:00
/*
2019-04-08 02:00:49 +00:00
* Copyright (c) 2018-2019 Atmosphère-NX
2019-02-20 12:52:11 +00:00
*
* This program is free software; you can redistribute it and/or modify it
* under the terms and conditions of the GNU General Public License,
* version 2, as published by the Free Software Foundation.
*
* This program is distributed in the hope 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 <http://www.gnu.org/licenses/>.
*/
#include <string.h>
#include <stdint.h>
#include <errno.h>
#include "gpio.h"
#include "utils.h"
2019-07-23 17:33:34 +00:00
static volatile tegra_gpio_bank_t *gpio_get_bank(uint32_t pin) {
2019-02-20 12:52:11 +00:00
volatile tegra_gpio_t *gpio = gpio_get_regs();
2019-07-23 17:33:34 +00:00
uint32_t bank_number = (pin >> GPIO_BANK_SHIFT);
2019-02-20 12:52:11 +00:00
return &gpio->bank[bank_number];
}
2019-07-23 17:33:34 +00:00
static volatile uint32_t gpio_get_port(uint32_t pin) {
return ((pin >> GPIO_PORT_SHIFT) & GPIO_PORT_MASK);
2019-02-20 12:52:11 +00:00
}
2019-07-23 17:33:34 +00:00
static volatile uint32_t gpio_get_mask(uint32_t pin) {
uint32_t pin_number = (pin & GPIO_PIN_MASK);
2019-02-20 12:52:11 +00:00
return (1 << pin_number);
}
2019-07-23 17:33:34 +00:00
static void gpio_simple_register_set(uint32_t pin, bool should_be_set, uint32_t offset) {
/* Retrieve the register set that corresponds to the given pin and offset. */
2019-02-20 12:52:11 +00:00
uintptr_t cluster_addr = (uintptr_t)gpio_get_bank(pin) + offset;
uint32_t *cluster = (uint32_t *)cluster_addr;
2019-07-23 17:33:34 +00:00
/* Figure out the offset into the cluster, and the mask to be used. */
2019-02-20 12:52:11 +00:00
uint32_t port = gpio_get_port(pin);
uint32_t mask = gpio_get_mask(pin);
2019-07-23 17:33:34 +00:00
/* Set or clear the bit, as appropriate. */
2019-02-20 12:52:11 +00:00
if (should_be_set)
2019-07-23 17:33:34 +00:00
cluster[port] |= mask;
2019-02-20 12:52:11 +00:00
else
cluster[port] &= ~mask;
2019-07-23 17:33:34 +00:00
/* Dummy read. */
(void)cluster[port];
2019-02-20 12:52:11 +00:00
}
2019-07-23 17:33:34 +00:00
static bool gpio_simple_register_get(uint32_t pin, uint32_t offset) {
/* Retrieve the register set that corresponds to the given pin and offset. */
2019-02-20 12:52:11 +00:00
uintptr_t cluster_addr = (uintptr_t)gpio_get_bank(pin) + offset;
uint32_t *cluster = (uint32_t *)cluster_addr;
2019-07-23 17:33:34 +00:00
/* Figure out the offset into the cluster, and the mask to be used. */
2019-02-20 12:52:11 +00:00
uint32_t port = gpio_get_port(pin);
uint32_t mask = gpio_get_mask(pin);
2019-07-23 17:33:34 +00:00
/* Convert the given value to a boolean. */
2019-02-20 12:52:11 +00:00
return !!(cluster[port] & mask);
}
2019-07-23 17:33:34 +00:00
void gpio_configure_mode(uint32_t pin, uint32_t mode) {
2019-02-20 12:52:11 +00:00
gpio_simple_register_set(pin, mode == GPIO_MODE_GPIO, offsetof(tegra_gpio_bank_t, config));
}
2019-07-23 17:33:34 +00:00
void gpio_configure_direction(uint32_t pin, uint32_t dir) {
2019-02-20 12:52:11 +00:00
gpio_simple_register_set(pin, dir == GPIO_DIRECTION_OUTPUT, offsetof(tegra_gpio_bank_t, direction));
}
2019-07-23 17:33:34 +00:00
void gpio_write(uint32_t pin, uint32_t value) {
2019-02-20 12:52:11 +00:00
gpio_simple_register_set(pin, value == GPIO_LEVEL_HIGH, offsetof(tegra_gpio_bank_t, out));
}
2019-07-23 17:33:34 +00:00
uint32_t gpio_read(uint32_t pin) {
2019-02-20 12:52:11 +00:00
return gpio_simple_register_get(pin, offsetof(tegra_gpio_bank_t, in));
}