blob: 9a976b8f1f6f1e7088eb8f3518513195654ff7e3 [file] [log] [blame]
/*
* Copyright (c) 2019, Nordic Semiconductor
*
* SPDX-License-Identifier: Apache-2.0
*/
#define DT_DRV_COMPAT nordic_nrf_ipc
#include <string.h>
#include <drivers/ipm.h>
#include <nrfx_ipc.h>
#include "ipm_nrfx_ipc.h"
#define LOG_LEVEL CONFIG_IPM_LOG_LEVEL
#include <logging/log.h>
LOG_MODULE_REGISTER(ipm_nrfx_ipc);
struct ipm_nrf_data {
ipm_callback_t callback;
void *callback_ctx;
};
static struct ipm_nrf_data nrfx_ipm_data;
static void gipm_init(void);
static void gipm_send(uint32_t id);
#if IS_ENABLED(CONFIG_IPM_NRF_SINGLE_INSTANCE)
static void nrfx_ipc_handler(uint32_t event_mask, void *p_context)
{
if (nrfx_ipm_data.callback) {
while (event_mask) {
uint8_t event_idx = __CLZ(__RBIT(event_mask));
__ASSERT(event_idx < NRFX_IPC_ID_MAX_VALUE,
"Illegal event_idx: %d", event_idx);
event_mask &= ~BIT(event_idx);
nrfx_ipm_data.callback(nrfx_ipm_data.callback_ctx,
event_idx,
NULL);
}
}
}
static int ipm_nrf_send(struct device *dev, int wait, uint32_t id,
const void *data, int size)
{
if (id > NRFX_IPC_ID_MAX_VALUE) {
return -EINVAL;
}
if (size > 0) {
LOG_WRN("nRF driver does not support sending data over IPM");
}
gipm_send(id);
return 0;
}
static int ipm_nrf_max_data_size_get(struct device *dev)
{
ARG_UNUSED(dev);
return 0;
}
static uint32_t ipm_nrf_max_id_val_get(struct device *dev)
{
ARG_UNUSED(dev);
return NRFX_IPC_ID_MAX_VALUE;
}
static void ipm_nrf_register_callback(struct device *dev,
ipm_callback_t cb,
void *context)
{
nrfx_ipm_data.callback = cb;
nrfx_ipm_data.callback_ctx = context;
}
static int ipm_nrf_set_enabled(struct device *dev, int enable)
{
/* Enable configured channels */
if (enable) {
irq_enable(DT_INST_IRQN(0));
nrfx_ipc_receive_event_group_enable((uint32_t)IPC_EVENT_BITS);
} else {
irq_disable(DT_INST_IRQN(0));
nrfx_ipc_receive_event_group_disable((uint32_t)IPC_EVENT_BITS);
}
return 0;
}
static int ipm_nrf_init(struct device *dev)
{
gipm_init();
return 0;
}
static const struct ipm_driver_api ipm_nrf_driver_api = {
.send = ipm_nrf_send,
.register_callback = ipm_nrf_register_callback,
.max_data_size_get = ipm_nrf_max_data_size_get,
.max_id_val_get = ipm_nrf_max_id_val_get,
.set_enabled = ipm_nrf_set_enabled
};
DEVICE_AND_API_INIT(ipm_nrf, DT_INST_LABEL(0),
ipm_nrf_init, NULL, NULL,
PRE_KERNEL_1, CONFIG_KERNEL_INIT_PRIORITY_DEFAULT,
&ipm_nrf_driver_api);
#else
struct vipm_nrf_data {
ipm_callback_t callback[NRFX_IPC_ID_MAX_VALUE];
void *callback_ctx[NRFX_IPC_ID_MAX_VALUE];
bool ipm_init;
struct device *ipm_device;
};
static struct vipm_nrf_data nrfx_vipm_data;
static void vipm_dispatcher(uint32_t event_mask, void *p_context)
{
while (event_mask) {
uint8_t event_idx = __CLZ(__RBIT(event_mask));
__ASSERT(event_idx < NRFX_IPC_ID_MAX_VALUE,
"Illegal event_idx: %d", event_idx);
event_mask &= ~BIT(event_idx);
if (nrfx_vipm_data.callback[event_idx] != NULL) {
nrfx_vipm_data.callback[event_idx]
(nrfx_vipm_data.callback_ctx[event_idx],
0,
NULL);
}
}
}
static int vipm_nrf_max_data_size_get(struct device *dev)
{
return ipm_max_data_size_get(dev);
}
static uint32_t vipm_nrf_max_id_val_get(struct device *dev)
{
ARG_UNUSED(dev);
return 0;
}
static int vipm_nrf_init(struct device *dev)
{
if (!nrfx_vipm_data.ipm_init) {
gipm_init();
nrfx_vipm_data.ipm_init = true;
}
return 0;
}
#define VIPM_DEVICE_1(_idx) \
static int vipm_nrf_##_idx##_send(struct device *dev, int wait, \
uint32_t id, const void *data, int size) \
{ \
if (!IS_ENABLED(CONFIG_IPM_MSG_CH_##_idx##_TX)) { \
LOG_ERR("IPM_" #_idx " is RX message channel"); \
return -EINVAL; \
} \
\
if (id > NRFX_IPC_ID_MAX_VALUE) { \
return -EINVAL; \
} \
\
if (id != 0) { \
LOG_WRN("Passing message ID to IPM with" \
"predefined message ID"); \
} \
\
if (size > 0) { \
LOG_WRN("nRF driver does not support" \
"sending data over IPM"); \
} \
\
gipm_send(_idx); \
return 0; \
} \
\
static void vipm_nrf_##_idx##_register_callback(struct device *dev, \
ipm_callback_t cb, \
void *context) \
{ \
if (IS_ENABLED(CONFIG_IPM_MSG_CH_##_idx##_RX)) { \
nrfx_vipm_data.callback[_idx] = cb; \
nrfx_vipm_data.callback_ctx[_idx] = context; \
} else { \
LOG_WRN("Trying to register a callback" \
"for TX channel IPM_" #_idx); \
} \
} \
\
static int vipm_nrf_##_idx##_set_enabled(struct device *dev, int enable)\
{ \
if (!IS_ENABLED(CONFIG_IPM_MSG_CH_##_idx##_RX)) { \
LOG_ERR("IPM_" #_idx " is TX message channel"); \
return -EINVAL; \
} else if (enable) { \
irq_enable(DT_INST_IRQN(0)); \
nrfx_ipc_receive_event_enable(_idx); \
} else if (!enable) { \
nrfx_ipc_receive_event_disable(_idx); \
} \
return 0; \
} \
\
static const struct ipm_driver_api vipm_nrf_##_idx##_driver_api = { \
.send = vipm_nrf_##_idx##_send, \
.register_callback = vipm_nrf_##_idx##_register_callback, \
.max_data_size_get = vipm_nrf_max_data_size_get, \
.max_id_val_get = vipm_nrf_max_id_val_get, \
.set_enabled = vipm_nrf_##_idx##_set_enabled \
}; \
\
DEVICE_AND_API_INIT(vipm_nrf_##_idx, "IPM_"#_idx, \
vipm_nrf_init, NULL, NULL, \
PRE_KERNEL_2, CONFIG_KERNEL_INIT_PRIORITY_DEFAULT, \
&vipm_nrf_##_idx##_driver_api)
#define VIPM_DEVICE(_idx, _) \
IF_ENABLED(CONFIG_IPM_MSG_CH_##_idx##_ENABLE, (VIPM_DEVICE_1(_idx);))
UTIL_LISTIFY(NRFX_IPC_ID_MAX_VALUE, VIPM_DEVICE, _);
#endif
static void gipm_init(void)
{
/* Init IPC */
#if IS_ENABLED(CONFIG_IPM_NRF_SINGLE_INSTANCE)
nrfx_ipc_init(0, nrfx_ipc_handler, (void *)&nrfx_ipm_data);
#else
nrfx_ipc_init(0, vipm_dispatcher, (void *)&nrfx_ipm_data);
#endif
IRQ_CONNECT(DT_INST_IRQN(0),
DT_INST_IRQ(0, priority),
nrfx_isr, nrfx_ipc_irq_handler, 0);
/* Set up signals and channels */
nrfx_ipc_config_load(&ipc_cfg);
}
static void gipm_send(uint32_t id)
{
nrfx_ipc_signal(id);
}