2017-03-31 10:41:27 +02:00
|
|
|
/*
|
|
|
|
* Copyright(C) 2017 Imagination Technologies Limited and/or its
|
|
|
|
* affiliated group companies.
|
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
|
|
* directory for more details.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2017-06-22 15:43:17 +02:00
|
|
|
/**
|
|
|
|
* @ingroup cpu_mips_pic32_common
|
|
|
|
* @ingroup drivers_periph_gpio
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Low-level GPIO driver implementation
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
2019-10-12 20:16:38 +02:00
|
|
|
#include "cpu.h"
|
2017-03-31 10:41:27 +02:00
|
|
|
#include "periph/gpio.h"
|
2019-10-12 20:16:38 +02:00
|
|
|
#include "periph_conf.h"
|
|
|
|
|
|
|
|
#define LATx(P) ((P)[0x30/0x4])
|
|
|
|
#define LATxCLR(P) ((P)[0x34/0x4])
|
|
|
|
#define LATxSET(P) ((P)[0x38/0x4])
|
|
|
|
#define LATxINV(P) ((P)[0x3C/0x4])
|
|
|
|
#define PORTx(P) ((P)[0x20/0x4])
|
|
|
|
#define CNPUxCLR(P) ((P)[0x54/0x4])
|
|
|
|
#define CNPUxSET(P) ((P)[0x58/0x4])
|
|
|
|
#define CNPDxCLR(P) ((P)[0x64/0x4])
|
|
|
|
#define CNPDxSET(P) ((P)[0x68/0x4])
|
|
|
|
#define ODCxCLR(P) ((P)[0x44/0x4])
|
|
|
|
#define ODCxSET(P) ((P)[0x48/0x4])
|
|
|
|
#define ANSELxCLR(P) ((P)[0x04/0x4])
|
|
|
|
#define TRISxCLR(P) ((P)[0x14/0x4])
|
|
|
|
#define TRISxSET(P) ((P)[0x18/0x4])
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Extract the port base address from the given pin identifier
|
|
|
|
*/
|
|
|
|
static inline volatile unsigned int * _port(gpio_t pin)
|
|
|
|
{
|
|
|
|
return (volatile unsigned int *)(pin & ~(0xff));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Extract the port number form the given identifier
|
|
|
|
*
|
|
|
|
* The port number is extracted by looking at bits 8, 9, 10, 11 of the base
|
|
|
|
* register addresses.
|
|
|
|
*/
|
|
|
|
static inline int _port_num(gpio_t pin)
|
|
|
|
{
|
|
|
|
return ((pin >> 8) & 0x0f);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Extract the pin number from the last 4 bit of the pin identifier
|
|
|
|
*/
|
|
|
|
static inline int _pin_num(gpio_t pin)
|
2017-03-31 10:41:27 +02:00
|
|
|
{
|
2019-10-12 20:16:38 +02:00
|
|
|
return (pin & 0x0f);
|
2017-03-31 10:41:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int gpio_init(gpio_t pin, gpio_mode_t mode)
|
|
|
|
{
|
2019-10-12 20:16:38 +02:00
|
|
|
volatile unsigned int * port = _port(pin);
|
|
|
|
int pin_num = _pin_num(pin);
|
2017-03-31 10:41:27 +02:00
|
|
|
uint8_t output = 0, pu = 0, pd = 0, od = 0;
|
|
|
|
|
|
|
|
switch (mode) {
|
|
|
|
case GPIO_IN:
|
|
|
|
break;
|
|
|
|
case GPIO_IN_PD:
|
|
|
|
pd = 1;
|
|
|
|
break;
|
|
|
|
case GPIO_IN_PU:
|
|
|
|
pu = 1;
|
|
|
|
break;
|
|
|
|
case GPIO_OD_PU:
|
|
|
|
pu = 1;
|
|
|
|
case GPIO_OD:
|
|
|
|
od = 1;
|
|
|
|
case GPIO_OUT:
|
|
|
|
output = 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-10-12 20:16:38 +02:00
|
|
|
ANSELxCLR(port) = 1U << pin_num; /* Configure GPIO as digital */
|
2017-03-31 10:41:27 +02:00
|
|
|
|
|
|
|
if (pu)
|
2019-10-12 20:16:38 +02:00
|
|
|
CNPUxSET(port) = 1U << pin_num;
|
2017-03-31 10:41:27 +02:00
|
|
|
else
|
2019-10-12 20:16:38 +02:00
|
|
|
CNPUxCLR(port) = 1U << pin_num;
|
2017-03-31 10:41:27 +02:00
|
|
|
|
|
|
|
if (pd)
|
2019-10-12 20:16:38 +02:00
|
|
|
CNPDxSET(port) = 1U << pin_num;
|
2017-03-31 10:41:27 +02:00
|
|
|
else
|
2019-10-12 20:16:38 +02:00
|
|
|
CNPDxCLR(port) = 1U << pin_num;
|
2017-03-31 10:41:27 +02:00
|
|
|
|
|
|
|
if (od)
|
2019-10-12 20:16:38 +02:00
|
|
|
ODCxSET(port) = 1U << pin_num;
|
2017-03-31 10:41:27 +02:00
|
|
|
else
|
2019-10-12 20:16:38 +02:00
|
|
|
ODCxCLR(port) = 1U << pin_num;
|
2017-03-31 10:41:27 +02:00
|
|
|
|
|
|
|
if (output)
|
2019-10-12 20:16:38 +02:00
|
|
|
TRISxCLR(port) = 1U << pin_num;
|
2017-03-31 10:41:27 +02:00
|
|
|
else
|
2019-10-12 20:16:38 +02:00
|
|
|
TRISxSET(port) = 1U << pin_num;
|
2017-03-31 10:41:27 +02:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int gpio_read(gpio_t pin)
|
|
|
|
{
|
2019-10-12 20:16:38 +02:00
|
|
|
return PORTx(_port(pin)) & (1U << _pin_num(pin));
|
2017-03-31 10:41:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void gpio_set(gpio_t pin)
|
|
|
|
{
|
2019-10-12 20:16:38 +02:00
|
|
|
LATxSET(_port(pin)) = 1U << _pin_num(pin);
|
2017-03-31 10:41:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void gpio_clear(gpio_t pin)
|
|
|
|
{
|
2019-10-12 20:16:38 +02:00
|
|
|
LATxCLR(_port(pin)) = 1U << _pin_num(pin);
|
2017-03-31 10:41:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void gpio_toggle(gpio_t pin)
|
|
|
|
{
|
2019-10-12 20:16:38 +02:00
|
|
|
LATxINV(_port(pin)) = 1U << _pin_num(pin);
|
2017-03-31 10:41:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void gpio_write(gpio_t pin, int value)
|
|
|
|
{
|
|
|
|
if (value)
|
|
|
|
gpio_set(pin);
|
|
|
|
else
|
|
|
|
gpio_clear(pin);
|
|
|
|
}
|