Newer
Older
mbed-os / targets / TARGET_TOSHIBA / TARGET_TMPM4KN / gpio_irq_api.c
/* mbed Microcontroller Library
 * Copyright(C) Toshiba Electronic Device Solutions Corporation 2021
 * SPDX-License-Identifier: Apache-2.0
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
#include "gpio_irq_api.h"
#include "mbed_error.h"
#include "PeripheralNames.h"
#include "pinmap.h"
#include "mbed_critical.h"
#include "txzp_gpio.h"

#define CHANNEL_NUM                     (22)
#define DISABLE                         (0)
#define ENABLE                          (1)
#define CLR_INT_FLAG                    (0xC0)

const PinMap PinMap_GPIO_IRQ[] = {
    {PA2, GPIO_IRQ_0, PIN_DATA(0, 0)},
    {PA4, GPIO_IRQ_1, PIN_DATA(0, 0)},
    {PC3, GPIO_IRQ_3, PIN_DATA(0, 0)},
    {PE3, GPIO_IRQ_4, PIN_DATA(0, 0)},
    {PE5, GPIO_IRQ_5, PIN_DATA(0, 0)},
    {PU1, GPIO_IRQ_7, PIN_DATA(0, 0)},
    {PU3, GPIO_IRQ_8, PIN_DATA(0, 0)},
    {PU6, GPIO_IRQ_9, PIN_DATA(0, 0)},
    {PC2, GPIO_IRQ_A, PIN_DATA(0, 0)},
    {PE4, GPIO_IRQ_B, PIN_DATA(0, 0)},
    {PU0, GPIO_IRQ_C, PIN_DATA(0, 0)},
    {PU5, GPIO_IRQ_D, PIN_DATA(0, 0)},
    {PA1, GPIO_IRQ_F, PIN_DATA(0, 0)},
    {PN1, GPIO_IRQ_10, PIN_DATA(0, 0)},
    {PD1, GPIO_IRQ_11, PIN_DATA(0, 0)},
    {PD5, GPIO_IRQ_12, PIN_DATA(0, 0)},
    {PG3, GPIO_IRQ_15, PIN_DATA(0, 0)},
    {NC,  NC,         0}
};


extern _gpio_t gpio_port_add;
static uintptr_t channel_contexts[CHANNEL_NUM] = {0};
static gpio_irq_handler hal_irq_handler[CHANNEL_NUM] = {NULL};
static CG_INTActiveState CurrentState;

static void CG_SetSTBYReleaseINTSrc(CG_INTSrc, CG_INTActiveState, uint8_t);
static void INT_IRQHandler(PinName, uint32_t);

void INT00_IRQHandler(void)
{
    INT_IRQHandler(PA2, 0);
}

void INT01_IRQHandler(void)
{
    INT_IRQHandler(PA4, 1);
}

void INT03_IRQHandler(void)
{
    INT_IRQHandler(PC3, 3);
}

void INT04_IRQHandler(void)
{
    INT_IRQHandler(PE3, 4);
}

void INT05_IRQHandler(void)
{
    INT_IRQHandler(PE5, 5);
}

void INT07_IRQHandler(void)
{
    INT_IRQHandler(PU1, 7);
}

void INT08_IRQHandler(void)
{
    INT_IRQHandler(PU3, 8);
}

void INT09_IRQHandler(void)
{
    INT_IRQHandler(PU6, 9);
}

void INT10_IRQHandler(void)
{
    INT_IRQHandler(PC2, 10);
}

void INT11_IRQHandler(void)
{
    INT_IRQHandler(PE4, 11);
}

void INT12_IRQHandler(void)
{
    INT_IRQHandler(PU0, 12);
}

void INT13_IRQHandler(void)
{
    INT_IRQHandler(PU5, 13);
}

void INT15_IRQHandler(void)
{
    INT_IRQHandler(PA1, 15);
}

void INT16_IRQHandler(void)
{
    INT_IRQHandler(PN1, 16);
}

void INT17_IRQHandler(void)
{
    INT_IRQHandler(PD1, 17);
}

void INT18_IRQHandler(void)
{
    INT_IRQHandler(PD5, 18);
}

void INT21_IRQHandler(void)
{
    INT_IRQHandler(PG3, 21);
}

int gpio_irq_init(gpio_irq_t *obj, PinName pin, gpio_irq_handler handler, uintptr_t context)
{
    // Get gpio interrupt ID
    obj->irq_id = pinmap_peripheral(pin, PinMap_GPIO_IRQ);
    core_util_critical_section_enter();
    // Get GPIO port and pin num
    obj->port    = (PortName)PIN_PORT(pin);
    obj->pin_num = PIN_POS(pin);
    // Set pin level as LOW
    gpio_write_bit(&gpio_port_add, obj->port, obj->pin_num, GPIO_Mode_DATA, 0);
    // Enable gpio interrupt function
    pinmap_pinout(pin, PinMap_GPIO_IRQ);
    // Get GPIO irq source
    obj->irq_src = (CG_INTSrc)obj->irq_id;
    // Save irq handler
    hal_irq_handler[obj->irq_src] = handler;
    // Save irq id
    channel_contexts[obj->irq_src] = context;
    // Initialize interrupt event as both edges detection
    obj->event = CG_INT_ACTIVE_STATE_BOTH_EDGES;
    // Clear gpio pending interrupt
    NVIC_ClearPendingIRQ((IRQn_Type)obj->irq_id);
    // Set interrupt event and enable INTx clear
    CG_SetSTBYReleaseINTSrc(obj->irq_src, (CG_INTActiveState)obj->event, ENABLE);
    core_util_critical_section_exit();

    return 0;
}

void gpio_irq_free(gpio_irq_t *obj)
{
    // Clear gpio_irq
    NVIC_ClearPendingIRQ((IRQn_Type)obj->irq_id);
    // Reset interrupt handler
    hal_irq_handler[obj->irq_src] = NULL;
    // Reset interrupt id
    channel_contexts[obj->irq_src] = 0;

    // Disable GPIO interrupt on obj
    gpio_irq_disable(obj);
}

void gpio_irq_set(gpio_irq_t *obj, gpio_irq_event event, uint32_t enable)
{
    // Disable GPIO interrupt on obj
    gpio_irq_disable(obj);

    if (enable) {
        // Get gpio interrupt event
        if (event == IRQ_RISE) {
            if ((obj->event == CG_INT_ACTIVE_STATE_FALLING) ||
                    (obj->event == CG_INT_ACTIVE_STATE_BOTH_EDGES)) {
                obj->event = CG_INT_ACTIVE_STATE_BOTH_EDGES;
            } else {
                obj->event = CG_INT_ACTIVE_STATE_RISING;
            }
        } else if (event == IRQ_FALL) {
            if ((obj->event == CG_INT_ACTIVE_STATE_RISING) ||
                    (obj->event == CG_INT_ACTIVE_STATE_BOTH_EDGES)) {
                obj->event = CG_INT_ACTIVE_STATE_BOTH_EDGES;
            } else {
                obj->event = CG_INT_ACTIVE_STATE_FALLING;
            }
        } else {
            error("Not supported event\n");
        }
    } else {
        // Get gpio interrupt event
        if (event == IRQ_RISE) {
            if ((obj->event == CG_INT_ACTIVE_STATE_RISING) ||
                    (obj->event == CG_INT_ACTIVE_STATE_INVALID)) {
                obj->event = CG_INT_ACTIVE_STATE_BOTH_EDGES;
            } else {
                obj->event = CG_INT_ACTIVE_STATE_FALLING;
            }
        } else if (event == IRQ_FALL) {
            if ((obj->event == CG_INT_ACTIVE_STATE_FALLING) ||
                    (obj->event == CG_INT_ACTIVE_STATE_INVALID)) {
                obj->event = CG_INT_ACTIVE_STATE_BOTH_EDGES;
            } else {
                obj->event = CG_INT_ACTIVE_STATE_RISING;
            }
        } else {
            error("Not supported event\n");
        }
    }

    CurrentState = obj->event;
    if (obj->event != CG_INT_ACTIVE_STATE_INVALID) {
        // Set interrupt event and enable INTx clear
        CG_SetSTBYReleaseINTSrc(obj->irq_src, (CG_INTActiveState)obj->event, ENABLE);
        gpio_write_bit(&gpio_port_add, (gpio_gr_t)obj->port, (gpio_num_t)obj->pin_num, GPIO_Mode_DATA, 0);
    } else {
        gpio_write_bit(&gpio_port_add, (gpio_gr_t)obj->port, (gpio_num_t)obj->pin_num, GPIO_Mode_DATA, 1);
    }

    // Clear interrupt request
    NVIC_ClearPendingIRQ((IRQn_Type)obj->irq_id);
    // Enable GPIO interrupt on obj
    gpio_irq_enable(obj);
}

void gpio_irq_enable(gpio_irq_t *obj)
{
    // Clear and Enable gpio_irq object
    NVIC_ClearPendingIRQ((IRQn_Type)obj->irq_id);
    NVIC_EnableIRQ((IRQn_Type)obj->irq_id);
}

void gpio_irq_disable(gpio_irq_t *obj)
{
    // Disable gpio_irq object
    NVIC_DisableIRQ((IRQn_Type)obj->irq_id);
}

static void INT_IRQHandler(PinName pin, uint32_t index)
{
    PortName port;
    uint8_t  pin_num;
    gpio_pinstate_t data = GPIO_PIN_RESET;
    pin_num              = PIN_POS(pin);
    port                 = (PortName)PIN_PORT(pin);

    // Clear interrupt request
    CG_SetSTBYReleaseINTSrc((CG_INTSrc)(CG_INT_SRC_0 + index), CurrentState, DISABLE);
    // Get pin value
    gpio_read_bit(&gpio_port_add, (gpio_gr_t)port, (gpio_num_t)pin_num, GPIO_Mode_DATA, &data);

    switch (data) {
        // Falling edge detection
        case 0:
            hal_irq_handler[index](channel_contexts[index], IRQ_FALL);
            break;
        // Rising edge detection
        case 1:
            hal_irq_handler[index](channel_contexts[index], IRQ_RISE);
            break;
        default:
            break;
    }

    // Clear gpio pending interrupt
    NVIC_ClearPendingIRQ((IRQn_Type)(CG_INT_SRC_0 + index));
    // Enable interrupt request
    CG_SetSTBYReleaseINTSrc((CG_INTSrc)(CG_INT_SRC_0 + index), CurrentState, ENABLE);
}

static void CG_SetSTBYReleaseINTSrc(CG_INTSrc INTSource, CG_INTActiveState ActiveState, uint8_t NewState)
{
    uint8_t *ptr = NULL;
    if (INTSource == 0) {
        ptr = (uint8_t *)(&(TSB_IB->IMC033));
        *ptr = CLR_INT_FLAG;
        *ptr = (ActiveState | NewState);
    } else if (INTSource <= 8) {
        ptr = ((uint8_t *)(&(TSB_IB->IMC034)) + ((INTSource - 1) * 2));
        *ptr = CLR_INT_FLAG;
        *ptr = (ActiveState | NewState);
    } else if (INTSource <= 10) {
        ptr = ((uint8_t *)(&(TSB_IB->IMC050)) + ((INTSource - 9)));
        *ptr = CLR_INT_FLAG;
        *ptr = (ActiveState | NewState);
    } else if (INTSource == 11) {
        ptr = (uint8_t *)(&(TSB_IB->IMC052));
        *ptr = CLR_INT_FLAG;
        *ptr = (ActiveState | NewState);
    } else if (INTSource <= 13) {
        ptr = ((uint8_t *)(&(TSB_IB->IMC054)) + ((INTSource - 12)));
        *ptr = CLR_INT_FLAG;
        *ptr = (ActiveState | NewState);
    } else if (INTSource == 14) {
        ptr = (uint8_t *)(&(TSB_IB->IMC056));
        *ptr = CLR_INT_FLAG;
        *ptr = (ActiveState | NewState);
    } else if (INTSource == 15) {
        ptr = (uint8_t *)(&(TSB_IB->IMC058));
        *ptr = CLR_INT_FLAG;
        *ptr = (ActiveState | NewState);
    } else if (INTSource <= 18) {
        ptr = ((uint8_t *)(&(TSB_IB->IMC059)) + ((INTSource - 16) * 2));
        *ptr = CLR_INT_FLAG;
        *ptr = (ActiveState | NewState);
    } else if (INTSource == 21) {
        ptr = (uint8_t *)(&(TSB_IB->IMC069));
        *ptr = CLR_INT_FLAG;
        *ptr = (ActiveState | NewState);
    }
    {
        uint8_t regval = *ptr;
        (void)regval;
    }
}