156 lines
4.2 KiB
C
156 lines
4.2 KiB
C
/*
|
|
* Copyright (c) 2017 Intel Corporation
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
#define DT_DRV_COMPAT intel_cavs_intc
|
|
|
|
#include <device.h>
|
|
#include <irq_nextlevel.h>
|
|
#include "intc_cavs.h"
|
|
|
|
#if defined(CONFIG_SMP) && (CONFIG_MP_NUM_CPUS > 1)
|
|
#if defined(CONFIG_SOC_INTEL_S1000)
|
|
#define PER_CPU_OFFSET(x) (0x40 * x)
|
|
#elif defined(CONFIG_SOC_INTEL_APL_ADSP)
|
|
#define PER_CPU_OFFSET(x) (0x40 * x)
|
|
#else
|
|
#error "Must define PER_CPU_OFFSET(x) for SoC"
|
|
#endif
|
|
#else
|
|
#define PER_CPU_OFFSET(x) 0
|
|
#endif
|
|
|
|
static ALWAYS_INLINE
|
|
struct cavs_registers *get_base_address(struct cavs_ictl_runtime *context)
|
|
{
|
|
#if defined(CONFIG_SMP) && (CONFIG_MP_NUM_CPUS > 1)
|
|
return UINT_TO_POINTER(context->base_addr +
|
|
PER_CPU_OFFSET(arch_curr_cpu()->id));
|
|
#else
|
|
return UINT_TO_POINTER(context->base_addr);
|
|
#endif
|
|
}
|
|
|
|
static ALWAYS_INLINE void cavs_ictl_dispatch_child_isrs(u32_t intr_status,
|
|
u32_t isr_base_offset)
|
|
{
|
|
u32_t intr_bitpos, intr_offset;
|
|
|
|
/* Dispatch lower level ISRs depending upon the bit set */
|
|
while (intr_status) {
|
|
intr_bitpos = find_lsb_set(intr_status) - 1;
|
|
intr_status &= ~(1 << intr_bitpos);
|
|
intr_offset = isr_base_offset + intr_bitpos;
|
|
_sw_isr_table[intr_offset].isr(
|
|
_sw_isr_table[intr_offset].arg);
|
|
}
|
|
}
|
|
|
|
static void cavs_ictl_isr(void *arg)
|
|
{
|
|
struct device *port = (struct device *)arg;
|
|
struct cavs_ictl_runtime *context = port->driver_data;
|
|
|
|
const struct cavs_ictl_config *config = port->config_info;
|
|
|
|
volatile struct cavs_registers * const regs =
|
|
get_base_address(context);
|
|
|
|
cavs_ictl_dispatch_child_isrs(regs->status_il,
|
|
config->isr_table_offset);
|
|
}
|
|
|
|
static inline void cavs_ictl_irq_enable(struct device *dev, unsigned int irq)
|
|
{
|
|
struct cavs_ictl_runtime *context = dev->driver_data;
|
|
|
|
volatile struct cavs_registers * const regs =
|
|
(struct cavs_registers *)context->base_addr;
|
|
|
|
regs->enable_il = (1 << irq);
|
|
}
|
|
|
|
static inline void cavs_ictl_irq_disable(struct device *dev, unsigned int irq)
|
|
{
|
|
struct cavs_ictl_runtime *context = dev->driver_data;
|
|
|
|
volatile struct cavs_registers * const regs =
|
|
(struct cavs_registers *)context->base_addr;
|
|
|
|
regs->disable_il = (1 << irq);
|
|
}
|
|
|
|
static inline unsigned int cavs_ictl_irq_get_state(struct device *dev)
|
|
{
|
|
struct cavs_ictl_runtime *context = dev->driver_data;
|
|
|
|
volatile struct cavs_registers * const regs =
|
|
(struct cavs_registers *)context->base_addr;
|
|
|
|
/* When the bits of this register are set, it means the
|
|
* corresponding interrupts are disabled. This function
|
|
* returns 0 only if ALL the interrupts are disabled.
|
|
*/
|
|
if (regs->disable_state_il == 0xFFFFFFFF) {
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int cavs_ictl_irq_get_line_state(struct device *dev, unsigned int irq)
|
|
{
|
|
struct cavs_ictl_runtime *context = dev->driver_data;
|
|
|
|
volatile struct cavs_registers * const regs =
|
|
(struct cavs_registers *)context->base_addr;
|
|
|
|
if ((regs->disable_state_il & BIT(irq)) == 0) {
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct irq_next_level_api cavs_apis = {
|
|
.intr_enable = cavs_ictl_irq_enable,
|
|
.intr_disable = cavs_ictl_irq_disable,
|
|
.intr_get_state = cavs_ictl_irq_get_state,
|
|
.intr_get_line_state = cavs_ictl_irq_get_line_state,
|
|
};
|
|
|
|
#define CAVS_ICTL_INIT(n) \
|
|
static int cavs_ictl_##n##_initialize(struct device *port) \
|
|
{ \
|
|
return 0; \
|
|
} \
|
|
\
|
|
static void cavs_config_##n##_irq(struct device *port); \
|
|
\
|
|
static const struct cavs_ictl_config cavs_config_##n = { \
|
|
.irq_num = DT_INST_IRQN(n), \
|
|
.isr_table_offset = CONFIG_CAVS_ISR_TBL_OFFSET + \
|
|
CONFIG_MAX_IRQ_PER_AGGREGATOR*n, \
|
|
.config_func = cavs_config_##n##_irq, \
|
|
}; \
|
|
\
|
|
static struct cavs_ictl_runtime cavs_##n##_runtime = { \
|
|
.base_addr = DT_INST_REG_ADDR(n), \
|
|
}; \
|
|
DEVICE_AND_API_INIT(cavs_ictl_##n, DT_INST_LABEL(n), \
|
|
cavs_ictl_##n##_initialize, \
|
|
&cavs_##n##_runtime, &cavs_config_##n, \
|
|
PRE_KERNEL_1, \
|
|
CONFIG_CAVS_ICTL_INIT_PRIORITY, &cavs_apis);\
|
|
\
|
|
static void cavs_config_##n##_irq(struct device *port) \
|
|
{ \
|
|
IRQ_CONNECT(DT_INST_IRQN(n), DT_INST_IRQ(n, priority), \
|
|
cavs_ictl_isr, DEVICE_GET(cavs_ictl_##n), \
|
|
DT_INST_IRQ(n, sense)); \
|
|
}
|
|
|
|
DT_INST_FOREACH_STATUS_OKAY(CAVS_ICTL_INIT)
|