1f96a0b5e9
Because call mutex lock is forbidden in interrupt. Signed-off-by: Bowen Wang <wangbowen6@xiaomi.com>
531 lines
13 KiB
C
531 lines
13 KiB
C
/****************************************************************************
|
|
* drivers/rpmsg/rpmsg.c
|
|
*
|
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
|
* contributor license agreements. See the NOTICE file distributed with
|
|
* this work for additional information regarding copyright ownership. The
|
|
* ASF licenses this file to you 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.
|
|
*
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
* Included Files
|
|
****************************************************************************/
|
|
|
|
#include <nuttx/config.h>
|
|
|
|
#include <metal/sys.h>
|
|
#include <nuttx/kmalloc.h>
|
|
#include <nuttx/mutex.h>
|
|
#include <nuttx/semaphore.h>
|
|
#include <nuttx/rpmsg/rpmsg.h>
|
|
|
|
#include "rpmsg_ping.h"
|
|
|
|
/****************************************************************************
|
|
* Private Types
|
|
****************************************************************************/
|
|
|
|
struct rpmsg_bind_s
|
|
{
|
|
char name[RPMSG_NAME_SIZE];
|
|
uint32_t dest;
|
|
struct metal_list node;
|
|
};
|
|
|
|
struct rpmsg_cb_s
|
|
{
|
|
FAR void *priv;
|
|
rpmsg_dev_cb_t device_created;
|
|
rpmsg_dev_cb_t device_destroy;
|
|
rpmsg_match_cb_t ns_match;
|
|
rpmsg_bind_cb_t ns_bind;
|
|
struct metal_list node;
|
|
};
|
|
|
|
/****************************************************************************
|
|
* Private Function Prototypes
|
|
****************************************************************************/
|
|
|
|
static int rpmsg_dev_ioctl(FAR struct file *filep, int cmd,
|
|
unsigned long arg);
|
|
|
|
/****************************************************************************
|
|
* Private Data
|
|
****************************************************************************/
|
|
|
|
static METAL_DECLARE_LIST(g_rpmsg_cb);
|
|
static METAL_DECLARE_LIST(g_rpmsg);
|
|
|
|
static rmutex_t g_rpmsg_lock = NXRMUTEX_INITIALIZER;
|
|
|
|
static const struct file_operations g_rpmsg_dev_ops =
|
|
{
|
|
NULL, /* open */
|
|
NULL, /* close */
|
|
NULL, /* read */
|
|
NULL, /* write */
|
|
NULL, /* seek */
|
|
rpmsg_dev_ioctl, /* ioctl */
|
|
};
|
|
|
|
/****************************************************************************
|
|
* Private Functions
|
|
****************************************************************************/
|
|
|
|
static inline FAR struct rpmsg_s *
|
|
rpmsg_get_by_rdev(FAR struct rpmsg_device *rdev)
|
|
{
|
|
if (!rdev)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
return metal_container_of(rdev, struct rpmsg_s, rdev);
|
|
}
|
|
|
|
static int rpmsg_dev_ioctl_(FAR struct rpmsg_s *rpmsg, int cmd,
|
|
unsigned long arg)
|
|
{
|
|
int ret = -ENOTTY;
|
|
|
|
switch (cmd)
|
|
{
|
|
case RPMSGIOC_PANIC:
|
|
if (rpmsg->ops->panic)
|
|
{
|
|
rpmsg->ops->panic(rpmsg);
|
|
ret = OK;
|
|
}
|
|
|
|
break;
|
|
case RPMSGIOC_DUMP:
|
|
if (rpmsg->ops->dump)
|
|
{
|
|
rpmsg->ops->dump(rpmsg);
|
|
ret = OK;
|
|
}
|
|
|
|
break;
|
|
#ifdef CONFIG_RPMSG_PING
|
|
case RPMSGIOC_PING:
|
|
ret = rpmsg_ping(&rpmsg->ping, (FAR const struct rpmsg_ping_s *)arg);
|
|
break;
|
|
#endif
|
|
default:
|
|
if (rpmsg->ops->ioctl)
|
|
{
|
|
ret = rpmsg->ops->ioctl(rpmsg, cmd, arg);
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int rpmsg_dev_ioctl(FAR struct file *filep, int cmd,
|
|
unsigned long arg)
|
|
{
|
|
FAR struct rpmsg_s *rpmsg =
|
|
(FAR struct rpmsg_s *)filep->f_inode->i_private;
|
|
|
|
return rpmsg_dev_ioctl_(rpmsg, cmd, arg);
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Public Functions
|
|
****************************************************************************/
|
|
|
|
int rpmsg_wait(FAR struct rpmsg_endpoint *ept, FAR sem_t *sem)
|
|
{
|
|
FAR struct rpmsg_s *rpmsg;
|
|
|
|
if (!ept || !sem)
|
|
{
|
|
return -EINVAL;
|
|
}
|
|
|
|
rpmsg = rpmsg_get_by_rdev(ept->rdev);
|
|
if (!rpmsg || !rpmsg->ops->wait)
|
|
{
|
|
return nxsem_wait_uninterruptible(sem);
|
|
}
|
|
|
|
return rpmsg->ops->wait(rpmsg, sem);
|
|
}
|
|
|
|
int rpmsg_post(FAR struct rpmsg_endpoint *ept, FAR sem_t *sem)
|
|
{
|
|
FAR struct rpmsg_s *rpmsg;
|
|
|
|
if (!ept || !sem)
|
|
{
|
|
return -EINVAL;
|
|
}
|
|
|
|
rpmsg = rpmsg_get_by_rdev(ept->rdev);
|
|
if (!rpmsg || !rpmsg->ops->post)
|
|
{
|
|
return nxsem_post(sem);
|
|
}
|
|
|
|
return rpmsg->ops->post(rpmsg, sem);
|
|
}
|
|
|
|
FAR const char *rpmsg_get_cpuname(FAR struct rpmsg_device *rdev)
|
|
{
|
|
FAR struct rpmsg_s *rpmsg = rpmsg_get_by_rdev(rdev);
|
|
return rpmsg ? rpmsg->ops->get_cpuname(rpmsg) : NULL;
|
|
}
|
|
|
|
int rpmsg_get_tx_buffer_size(FAR struct rpmsg_device *rdev)
|
|
{
|
|
FAR struct rpmsg_s *rpmsg = rpmsg_get_by_rdev(rdev);
|
|
return rpmsg ? rpmsg->ops->get_tx_buffer_size(rpmsg) : -EINVAL;
|
|
}
|
|
|
|
int rpmsg_get_rx_buffer_size(FAR struct rpmsg_device *rdev)
|
|
{
|
|
FAR struct rpmsg_s *rpmsg = rpmsg_get_by_rdev(rdev);
|
|
return rpmsg ? rpmsg->ops->get_rx_buffer_size(rpmsg) : -EINVAL;
|
|
}
|
|
|
|
int rpmsg_register_callback(FAR void *priv,
|
|
rpmsg_dev_cb_t device_created,
|
|
rpmsg_dev_cb_t device_destroy,
|
|
rpmsg_match_cb_t ns_match,
|
|
rpmsg_bind_cb_t ns_bind)
|
|
{
|
|
FAR struct metal_list *node;
|
|
FAR struct metal_list *bnode;
|
|
FAR struct rpmsg_cb_s *cb;
|
|
|
|
cb = kmm_zalloc(sizeof(struct rpmsg_cb_s));
|
|
if (cb == NULL)
|
|
{
|
|
return -ENOMEM;
|
|
}
|
|
|
|
cb->priv = priv;
|
|
cb->device_created = device_created;
|
|
cb->device_destroy = device_destroy;
|
|
cb->ns_match = ns_match;
|
|
cb->ns_bind = ns_bind;
|
|
|
|
nxrmutex_lock(&g_rpmsg_lock);
|
|
metal_list_for_each(&g_rpmsg, node)
|
|
{
|
|
FAR struct rpmsg_s *rpmsg =
|
|
metal_container_of(node, struct rpmsg_s, node);
|
|
|
|
if (rpmsg->rdev->ns_unbind_cb == NULL)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if (device_created)
|
|
{
|
|
device_created(rpmsg->rdev, priv);
|
|
}
|
|
|
|
if (ns_bind == NULL)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
DEBUGASSERT(ns_match != NULL);
|
|
|
|
again:
|
|
nxrmutex_lock(&rpmsg->lock);
|
|
metal_list_for_each(&rpmsg->bind, bnode)
|
|
{
|
|
FAR struct rpmsg_bind_s *bind =
|
|
metal_container_of(bnode, struct rpmsg_bind_s, node);
|
|
|
|
if (ns_match(rpmsg->rdev, priv, bind->name, bind->dest))
|
|
{
|
|
metal_list_del(bnode);
|
|
nxrmutex_unlock(&rpmsg->lock);
|
|
ns_bind(rpmsg->rdev, priv, bind->name, bind->dest);
|
|
|
|
kmm_free(bind);
|
|
goto again;
|
|
}
|
|
}
|
|
|
|
nxrmutex_unlock(&rpmsg->lock);
|
|
}
|
|
|
|
metal_list_add_tail(&g_rpmsg_cb, &cb->node);
|
|
nxrmutex_unlock(&g_rpmsg_lock);
|
|
|
|
return 0;
|
|
}
|
|
|
|
void rpmsg_unregister_callback(FAR void *priv,
|
|
rpmsg_dev_cb_t device_created,
|
|
rpmsg_dev_cb_t device_destroy,
|
|
rpmsg_match_cb_t ns_match,
|
|
rpmsg_bind_cb_t ns_bind)
|
|
{
|
|
FAR struct metal_list *node;
|
|
FAR struct metal_list *pnode;
|
|
|
|
nxrmutex_lock(&g_rpmsg_lock);
|
|
metal_list_for_each(&g_rpmsg_cb, node)
|
|
{
|
|
FAR struct rpmsg_cb_s *cb =
|
|
metal_container_of(node, struct rpmsg_cb_s, node);
|
|
|
|
if (cb->priv == priv &&
|
|
cb->device_created == device_created &&
|
|
cb->device_destroy == device_destroy &&
|
|
cb->ns_match == ns_match &&
|
|
cb->ns_bind == ns_bind)
|
|
{
|
|
metal_list_del(&cb->node);
|
|
kmm_free(cb);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (device_destroy)
|
|
{
|
|
metal_list_for_each(&g_rpmsg, pnode)
|
|
{
|
|
FAR struct rpmsg_s *rpmsg =
|
|
metal_container_of(pnode, struct rpmsg_s, node);
|
|
|
|
if (rpmsg->rdev->ns_unbind_cb)
|
|
{
|
|
device_destroy(rpmsg->rdev, priv);
|
|
}
|
|
}
|
|
}
|
|
|
|
nxrmutex_unlock(&g_rpmsg_lock);
|
|
}
|
|
|
|
void rpmsg_ns_bind(FAR struct rpmsg_device *rdev,
|
|
FAR const char *name, uint32_t dest)
|
|
{
|
|
FAR struct rpmsg_s *rpmsg = rpmsg_get_by_rdev(rdev);
|
|
FAR struct rpmsg_bind_s *bind;
|
|
FAR struct metal_list *node;
|
|
|
|
nxrmutex_lock(&g_rpmsg_lock);
|
|
metal_list_for_each(&g_rpmsg_cb, node)
|
|
{
|
|
FAR struct rpmsg_cb_s *cb =
|
|
metal_container_of(node, struct rpmsg_cb_s, node);
|
|
|
|
if (cb->ns_match && cb->ns_match(rdev, cb->priv, name, dest))
|
|
{
|
|
rpmsg_bind_cb_t ns_bind = cb->ns_bind;
|
|
FAR void *cb_priv = cb->priv;
|
|
|
|
nxrmutex_unlock(&g_rpmsg_lock);
|
|
DEBUGASSERT(ns_bind != NULL);
|
|
ns_bind(rdev, cb_priv, name, dest);
|
|
return;
|
|
}
|
|
}
|
|
|
|
nxrmutex_unlock(&g_rpmsg_lock);
|
|
|
|
bind = kmm_malloc(sizeof(struct rpmsg_bind_s));
|
|
if (bind == NULL)
|
|
{
|
|
return;
|
|
}
|
|
|
|
bind->dest = dest;
|
|
strlcpy(bind->name, name, RPMSG_NAME_SIZE);
|
|
|
|
nxrmutex_lock(&rpmsg->lock);
|
|
metal_list_add_tail(&rpmsg->bind, &bind->node);
|
|
nxrmutex_unlock(&rpmsg->lock);
|
|
}
|
|
|
|
void rpmsg_ns_unbind(FAR struct rpmsg_device *rdev,
|
|
FAR const char *name, uint32_t dest)
|
|
{
|
|
FAR struct rpmsg_s *rpmsg = rpmsg_get_by_rdev(rdev);
|
|
FAR struct metal_list *node;
|
|
|
|
nxrmutex_lock(&rpmsg->lock);
|
|
metal_list_for_each(&rpmsg->bind, node)
|
|
{
|
|
FAR struct rpmsg_bind_s *bind =
|
|
metal_container_of(node, struct rpmsg_bind_s, node);
|
|
|
|
if (bind->dest == dest && !strncmp(bind->name, name, RPMSG_NAME_SIZE))
|
|
{
|
|
metal_list_del(node);
|
|
kmm_free(bind);
|
|
break;
|
|
}
|
|
}
|
|
|
|
nxrmutex_unlock(&rpmsg->lock);
|
|
}
|
|
|
|
void rpmsg_device_created(FAR struct rpmsg_s *rpmsg)
|
|
{
|
|
FAR struct metal_list *node;
|
|
|
|
nxrmutex_lock(&g_rpmsg_lock);
|
|
metal_list_for_each(&g_rpmsg_cb, node)
|
|
{
|
|
FAR struct rpmsg_cb_s *cb =
|
|
metal_container_of(node, struct rpmsg_cb_s, node);
|
|
|
|
if (cb->device_created)
|
|
{
|
|
cb->device_created(rpmsg->rdev, cb->priv);
|
|
}
|
|
}
|
|
|
|
nxrmutex_unlock(&g_rpmsg_lock);
|
|
|
|
#ifdef CONFIG_RPMSG_PING
|
|
rpmsg_ping_init(rpmsg->rdev, &rpmsg->ping);
|
|
#endif
|
|
}
|
|
|
|
void rpmsg_device_destory(FAR struct rpmsg_s *rpmsg)
|
|
{
|
|
FAR struct metal_list *node;
|
|
FAR struct metal_list *tmp;
|
|
|
|
#ifdef CONFIG_RPMSG_PING
|
|
rpmsg_ping_deinit(&rpmsg->ping);
|
|
#endif
|
|
|
|
nxrmutex_lock(&rpmsg->lock);
|
|
metal_list_for_each_safe(&rpmsg->bind, node, tmp)
|
|
{
|
|
FAR struct rpmsg_bind_s *bind =
|
|
metal_container_of(node, struct rpmsg_bind_s, node);
|
|
|
|
metal_list_del(node);
|
|
kmm_free(bind);
|
|
}
|
|
|
|
nxrmutex_unlock(&rpmsg->lock);
|
|
|
|
/* Broadcast device_destroy to all registers */
|
|
|
|
nxrmutex_lock(&g_rpmsg_lock);
|
|
metal_list_for_each(&g_rpmsg_cb, node)
|
|
{
|
|
FAR struct rpmsg_cb_s *cb =
|
|
metal_container_of(node, struct rpmsg_cb_s, node);
|
|
|
|
if (cb->device_destroy)
|
|
{
|
|
cb->device_destroy(rpmsg->rdev, cb->priv);
|
|
}
|
|
}
|
|
|
|
nxrmutex_unlock(&g_rpmsg_lock);
|
|
}
|
|
|
|
int rpmsg_register(FAR const char *path, FAR struct rpmsg_s *rpmsg,
|
|
FAR const struct rpmsg_ops_s *ops)
|
|
{
|
|
struct metal_init_params params = METAL_INIT_DEFAULTS;
|
|
int ret;
|
|
|
|
ret = metal_init(¶ms);
|
|
if (ret < 0)
|
|
{
|
|
return ret;
|
|
}
|
|
|
|
ret = register_driver(path, &g_rpmsg_dev_ops, 0222, rpmsg);
|
|
if (ret < 0)
|
|
{
|
|
metal_finish();
|
|
return ret;
|
|
}
|
|
|
|
metal_list_init(&rpmsg->bind);
|
|
nxrmutex_init(&rpmsg->lock);
|
|
rpmsg->ops = ops;
|
|
|
|
/* Add priv to list */
|
|
|
|
nxrmutex_lock(&g_rpmsg_lock);
|
|
metal_list_add_tail(&g_rpmsg, &rpmsg->node);
|
|
nxrmutex_unlock(&g_rpmsg_lock);
|
|
|
|
return ret;
|
|
}
|
|
|
|
void rpmsg_unregister(FAR const char *path, FAR struct rpmsg_s *rpmsg)
|
|
{
|
|
nxrmutex_lock(&g_rpmsg_lock);
|
|
metal_list_del(&rpmsg->node);
|
|
nxrmutex_unlock(&g_rpmsg_lock);
|
|
|
|
nxrmutex_destroy(&rpmsg->lock);
|
|
unregister_driver(path);
|
|
|
|
metal_finish();
|
|
}
|
|
|
|
int rpmsg_ioctl(FAR const char *cpuname, int cmd, unsigned long arg)
|
|
{
|
|
FAR struct metal_list *node;
|
|
int ret = OK;
|
|
|
|
if (!up_interrupt_context())
|
|
{
|
|
nxrmutex_lock(&g_rpmsg_lock);
|
|
}
|
|
|
|
metal_list_for_each(&g_rpmsg, node)
|
|
{
|
|
FAR struct rpmsg_s *rpmsg =
|
|
metal_container_of(node, struct rpmsg_s, node);
|
|
|
|
if (!cpuname || !strcmp(rpmsg_get_cpuname(rpmsg->rdev), cpuname))
|
|
{
|
|
ret = rpmsg_dev_ioctl_(rpmsg, cmd, arg);
|
|
if (ret < 0)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!up_interrupt_context())
|
|
{
|
|
nxrmutex_unlock(&g_rpmsg_lock);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
int rpmsg_panic(FAR const char *cpuname)
|
|
{
|
|
return rpmsg_ioctl(cpuname, RPMSGIOC_PANIC, 0);
|
|
}
|
|
|
|
void rpmsg_dump_all(void)
|
|
{
|
|
rpmsg_ioctl(NULL, RPMSGIOC_DUMP, 0);
|
|
}
|