/**************************************************************************** * drivers/rpmsg/rpmsg_port.c * * SPDX-License-Identifier: Apache-2.0 * * 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 #include #include #include #include #include #include "rpmsg_port.h" /**************************************************************************** * Pre-processor Definitions ****************************************************************************/ #define RPMSG_PORT_BUF_TO_NODE(q,b) ((q)->node + ((FAR void *)(b) - (q)->buf) / (q)->len) #define RPMSG_PORT_NODE_TO_BUF(q,n) ((q)->buf + (((n) - (q)->node)) * (q)->len) /**************************************************************************** * Private Function Prototypes ****************************************************************************/ static FAR const char * rpmsg_port_get_local_cpuname(FAR struct rpmsg_s *rpmsg); static FAR const char *rpmsg_port_get_cpuname(FAR struct rpmsg_s *rpmsg); static void rpmsg_port_dump(FAR struct rpmsg_s *rpmsg); /**************************************************************************** * Private Data ****************************************************************************/ static const struct rpmsg_ops_s g_rpmsg_port_ops = { NULL, NULL, NULL, NULL, rpmsg_port_dump, rpmsg_port_get_local_cpuname, rpmsg_port_get_cpuname, }; /**************************************************************************** * Private Functions ****************************************************************************/ /**************************************************************************** * Name: rpmsg_port_post ****************************************************************************/ static void rpmsg_port_post(FAR sem_t *sem) { int count = 0; nxsem_get_value(sem, &count); while (count++ <= 0) { nxsem_post(sem); } } /**************************************************************************** * Name: rpmsg_port_add_node ****************************************************************************/ static void rpmsg_port_add_node(FAR struct rpmsg_port_list_s *list, FAR struct list_node *node) { irqstate_t flags; flags = spin_lock_irqsave(&list->lock); list_add_tail(&list->head, node); list->num++; spin_unlock_irqrestore(&list->lock, flags); } /**************************************************************************** * Name: rpmsg_port_remove_node ****************************************************************************/ static FAR struct list_node * rpmsg_port_remove_node(FAR struct rpmsg_port_list_s *list) { FAR struct list_node *node; irqstate_t flags; flags = spin_lock_irqsave(&list->lock); node = list_remove_head(&list->head); if (node != NULL) { list->num--; } spin_unlock_irqrestore(&list->lock, flags); return node; } /**************************************************************************** * Name: rpmsg_port_destroy_queue * * Description: * Free memory alloced by rpmsg_port_create_queue. * ****************************************************************************/ static void rpmsg_port_destroy_queue(FAR struct rpmsg_port_queue_s *queue) { if (queue->alloced) { kmm_free(queue->buf); } kmm_free(queue->node); nxsem_destroy(&queue->free.sem); nxsem_destroy(&queue->ready.sem); } /**************************************************************************** * Name: rpmsg_port_create_queue ****************************************************************************/ static int rpmsg_port_create_queue(FAR struct rpmsg_port_queue_s *queue, uint16_t count, uint16_t len, FAR void *buf) { FAR struct list_node *node; node = kmm_malloc(count * sizeof(struct list_node)); if (node == NULL) { return -ENOMEM; } queue->node = node; /* Check if buffer space needed to be malloced internal. */ if (buf == NULL) { buf = kmm_malloc(count * len); if (buf == NULL) { kmm_free(queue->node); return -ENOMEM; } queue->alloced = true; } queue->len = len; queue->buf = buf; /* Init free list */ spin_lock_init(&queue->free.lock); nxsem_init(&queue->free.sem, 0, 0); list_initialize(&queue->free.head); while (count--) { rpmsg_port_add_node(&queue->free, node); node++; } /* Init ready list */ spin_lock_init(&queue->ready.lock); nxsem_init(&queue->ready.sem, 0, 0); list_initialize(&queue->ready.head); return 0; } /**************************************************************************** * Name: rpmsg_port_create_queues ****************************************************************************/ static int rpmsg_port_create_queues(FAR struct rpmsg_port_s *port, FAR const struct rpmsg_port_config_s *cfg) { int ret; ret = rpmsg_port_create_queue(&port->txq, cfg->txnum, cfg->txlen, cfg->txbuf); if (ret < 0) { return ret; } ret = rpmsg_port_create_queue(&port->rxq, cfg->rxnum, cfg->rxlen, cfg->rxbuf); if (ret < 0) { rpmsg_port_destroy_queue(&port->txq); return ret; } return 0; } /**************************************************************************** * Name: rpmsg_port_destroy_queues ****************************************************************************/ static void rpmsg_port_destroy_queues(FAR struct rpmsg_port_s *port) { rpmsg_port_destroy_queue(&port->txq); rpmsg_port_destroy_queue(&port->rxq); } /**************************************************************************** * Name: rpmsg_port_get_tx_payload_buffer ****************************************************************************/ static FAR void * rpmsg_port_get_tx_payload_buffer(FAR struct rpmsg_device *rdev, FAR uint32_t *len, int wait) { FAR struct rpmsg_port_s *port = metal_container_of(rdev, struct rpmsg_port_s, rdev); FAR struct rpmsg_port_header_s *hdr = rpmsg_port_queue_get_available_buffer(&port->txq, wait); if (hdr == NULL) { return NULL; } *len = hdr->len - sizeof(struct rpmsg_port_header_s) - sizeof(struct rpmsg_hdr); return RPMSG_LOCATE_DATA(hdr->buf); } /**************************************************************************** * Name: rpmsg_port_send_offchannel_nocopy ****************************************************************************/ static int rpmsg_port_send_offchannel_nocopy(FAR struct rpmsg_device *rdev, uint32_t src, uint32_t dst, FAR const void *data, int len) { FAR struct rpmsg_port_s *port = metal_container_of(rdev, struct rpmsg_port_s, rdev); FAR struct rpmsg_port_header_s *hdr; FAR struct rpmsg_hdr *rphdr; rphdr = RPMSG_LOCATE_HDR(data); rphdr->dst = dst; rphdr->src = src; rphdr->len = len; rphdr->reserved = 0; rphdr->flags = 0; hdr = metal_container_of(rphdr, struct rpmsg_port_header_s, buf); hdr->len = sizeof(struct rpmsg_port_header_s) + sizeof(struct rpmsg_hdr) + len; rpmsg_port_queue_add_buffer(&port->txq, hdr); if (port->ops->notify_tx_ready) { port->ops->notify_tx_ready(port); } return len; } /**************************************************************************** * Name: rpmsg_port_send_offchannel_raw ****************************************************************************/ static int rpmsg_port_send_offchannel_raw(FAR struct rpmsg_device *rdev, uint32_t src, uint32_t dst, FAR const void *data, int len, int wait) { uint32_t buflen; FAR void *buf; buf = rpmsg_port_get_tx_payload_buffer(rdev, &buflen, wait); if (buf == NULL) { return RPMSG_ERR_NO_BUFF; } RPMSG_ASSERT(len <= buflen, "Send size larger than buffer size\n"); memcpy(buf, data, len); return rpmsg_port_send_offchannel_nocopy(rdev, src, dst, buf, len); } /**************************************************************************** * Name: rpmsg_port_hold_rx_buffer ****************************************************************************/ static void rpmsg_port_hold_rx_buffer(FAR struct rpmsg_device *rdev, FAR void *rxbuf) { FAR struct rpmsg_hdr *rphdr = RPMSG_LOCATE_HDR(rxbuf); atomic_fetch_add(&rphdr->reserved, 1 << RPMSG_BUF_HELD_SHIFT); } /**************************************************************************** * Name: rpmsg_port_release_rx_buffer ****************************************************************************/ static void rpmsg_port_release_rx_buffer(FAR struct rpmsg_device *rdev, FAR void *rxbuf) { FAR struct rpmsg_port_s *port = metal_container_of(rdev, struct rpmsg_port_s, rdev); FAR struct rpmsg_hdr *rphdr = RPMSG_LOCATE_HDR(rxbuf); FAR struct rpmsg_port_header_s *hdr = metal_container_of(rphdr, struct rpmsg_port_header_s, buf); uint32_t reserved = atomic_fetch_sub(&rphdr->reserved, 1 << RPMSG_BUF_HELD_SHIFT); if ((reserved & RPMSG_BUF_HELD_MASK) == (1 << RPMSG_BUF_HELD_SHIFT)) { rpmsg_port_queue_return_buffer(&port->rxq, hdr); if (port->ops->notify_rx_free) { port->ops->notify_rx_free(port); } } } /**************************************************************************** * Name: rpmsg_port_release_tx_buffer ****************************************************************************/ static int rpmsg_port_release_tx_buffer(FAR struct rpmsg_device *rdev, FAR void *txbuf) { FAR struct rpmsg_port_s *port = metal_container_of(rdev, struct rpmsg_port_s, rdev); FAR struct rpmsg_hdr *rphdr = RPMSG_LOCATE_HDR(txbuf); FAR struct rpmsg_port_header_s *hdr = metal_container_of(rphdr, struct rpmsg_port_header_s, buf); rpmsg_port_queue_return_buffer(&port->txq, hdr); return RPMSG_SUCCESS; } /**************************************************************************** * Name: rpmsg_port_get_tx_buffer_size ****************************************************************************/ static int rpmsg_port_get_tx_buffer_size(FAR struct rpmsg_device *rdev) { FAR struct rpmsg_port_s *port = metal_container_of(rdev, struct rpmsg_port_s, rdev); return port->txq.len - sizeof(struct rpmsg_port_header_s) - sizeof(struct rpmsg_hdr); } /**************************************************************************** * Name: rpmsg_port_get_rx_buffer_size ****************************************************************************/ static int rpmsg_port_get_rx_buffer_size(FAR struct rpmsg_device *rdev) { FAR struct rpmsg_port_s *port = metal_container_of(rdev, struct rpmsg_port_s, rdev); return port->rxq.len - sizeof(struct rpmsg_port_header_s) - sizeof(struct rpmsg_hdr); } /**************************************************************************** * Name: rpmsg_port_rx_callback ****************************************************************************/ static void rpmsg_port_rx_callback(FAR struct rpmsg_port_s *port, FAR struct rpmsg_port_header_s *hdr) { FAR struct rpmsg_device *rdev = &port->rdev; FAR struct rpmsg_hdr *rphdr = (FAR struct rpmsg_hdr *)hdr->buf; FAR void *data = RPMSG_LOCATE_DATA(rphdr); FAR struct rpmsg_endpoint *ept; int status; metal_mutex_acquire(&rdev->lock); ept = rpmsg_get_ept_from_addr(rdev, rphdr->dst); rpmsg_ept_incref(ept); metal_mutex_release(&rdev->lock); rpmsg_port_hold_rx_buffer(rdev, data); if (ept != NULL) { if (ept->dest_addr == RPMSG_ADDR_ANY) { ept->dest_addr = rphdr->src; } status = ept->cb(ept, data, rphdr->len, rphdr->src, ept->priv); if (status < 0) { RPMSG_ASSERT(0, "unexpected callback status\n"); } } rpmsg_port_release_rx_buffer(rdev, data); metal_mutex_acquire(&rdev->lock); rpmsg_ept_decref(ept); metal_mutex_release(&rdev->lock); } /**************************************************************************** * Name: rpmsg_port_ns_callback ****************************************************************************/ static int rpmsg_port_ns_callback(FAR struct rpmsg_endpoint *ept, FAR void *data, size_t len, uint32_t src, FAR void *priv) { FAR struct rpmsg_device *rdev = ept->rdev; FAR struct rpmsg_ns_msg *msg = data; FAR const char *name = msg->name; uint32_t dest = msg->addr; bool decref = false; if (len != sizeof(*msg)) { return RPMSG_SUCCESS; } metal_mutex_acquire(&rdev->lock); ept = rpmsg_get_endpoint(rdev, name, RPMSG_ADDR_ANY, dest); if (msg->flags == RPMSG_NS_DESTROY) { if (ept != NULL) { ept->dest_addr = RPMSG_ADDR_ANY; if (ept->release_cb != NULL) { rpmsg_ept_incref(ept); decref = true; } } metal_mutex_release(&rdev->lock); if (ept != NULL && ept->ns_unbind_cb != NULL) { ept->ns_unbind_cb(ept); } if (rdev->ns_unbind_cb != NULL) { rdev->ns_unbind_cb(rdev, name, dest); } if (decref) { metal_mutex_acquire(&rdev->lock); rpmsg_ept_decref(ept); metal_mutex_release(&rdev->lock); } } else if (msg->flags == RPMSG_NS_CREATE) { if (ept == NULL) { metal_mutex_release(&rdev->lock); if (rdev->ns_bind_cb != NULL) { rdev->ns_bind_cb(rdev, name, dest); } } else if (ept->dest_addr == RPMSG_ADDR_ANY) { ept->dest_addr = dest; metal_mutex_release(&rdev->lock); if (ept->name[0] && rdev->support_ack) { rpmsg_send_ns_message(ept, RPMSG_NS_CREATE_ACK); } /* Notify application that the endpoint has been bound */ if (ept->ns_bound_cb != NULL) { ept->ns_bound_cb(ept); } } else { metal_mutex_release(&rdev->lock); } } else { /* RPMSG_NS_CREATE_ACK */ if (ept != NULL && ept->dest_addr == RPMSG_ADDR_ANY) { ept->dest_addr = dest; metal_mutex_release(&rdev->lock); if (ept->ns_bound_cb != NULL) { ept->ns_bound_cb(ept); } } else { metal_mutex_release(&rdev->lock); } } return RPMSG_SUCCESS; } /**************************************************************************** * Name: rpmsg_port_get_local_cpuname ****************************************************************************/ static FAR const char * rpmsg_port_get_local_cpuname(FAR struct rpmsg_s *rpmsg) { FAR struct rpmsg_port_s *port = (FAR struct rpmsg_port_s *)rpmsg; return port->local_cpuname; } /**************************************************************************** * Name: rpmsg_port_get_cpuname ****************************************************************************/ static FAR const char *rpmsg_port_get_cpuname(FAR struct rpmsg_s *rpmsg) { FAR struct rpmsg_port_s *port = (FAR struct rpmsg_port_s *)rpmsg; return port->cpuname; } /**************************************************************************** * Public Functions ****************************************************************************/ /**************************************************************************** * Name: rpmsg_port_initialize ****************************************************************************/ int rpmsg_port_initialize(FAR struct rpmsg_port_s *port, FAR const struct rpmsg_port_config_s *cfg, FAR const struct rpmsg_port_ops_s *ops) { FAR struct rpmsg_device *rdev; int ret; if (port == NULL || cfg == NULL || ops == NULL) { return -EINVAL; } ret = rpmsg_port_create_queues(port, cfg); if (ret < 0) { return ret; } port->ops = ops; strlcpy(port->cpuname, cfg->remotecpu, RPMSG_NAME_SIZE); rdev = &port->rdev; memset(rdev, 0, sizeof(*rdev)); metal_mutex_init(&rdev->lock); rdev->ns_bind_cb = rpmsg_ns_bind; rdev->ns_unbind_cb = rpmsg_ns_unbind; rdev->ops.send_offchannel_raw = rpmsg_port_send_offchannel_raw; rdev->ops.hold_rx_buffer = rpmsg_port_hold_rx_buffer; rdev->ops.release_rx_buffer = rpmsg_port_release_rx_buffer; rdev->ops.get_tx_payload_buffer = rpmsg_port_get_tx_payload_buffer; rdev->ops.send_offchannel_nocopy = rpmsg_port_send_offchannel_nocopy; rdev->ops.release_tx_buffer = rpmsg_port_release_tx_buffer; rdev->ops.get_rx_buffer_size = rpmsg_port_get_rx_buffer_size; rdev->ops.get_tx_buffer_size = rpmsg_port_get_tx_buffer_size; metal_list_init(&rdev->endpoints); rdev->support_ack = true; rdev->support_ns = true; rpmsg_register_endpoint(rdev, &rdev->ns_ept, "NS", RPMSG_NS_EPT_ADDR, RPMSG_NS_EPT_ADDR, rpmsg_port_ns_callback, NULL, port); port->ops->register_callback(port, rpmsg_port_rx_callback); return 0; } /**************************************************************************** * Name: rpmsg_port_uninitialize ****************************************************************************/ void rpmsg_port_uninitialize(FAR struct rpmsg_port_s *port) { FAR struct rpmsg_device *rdev = &port->rdev; FAR struct metal_list *node; FAR struct rpmsg_endpoint *ept; while (!metal_list_is_empty(&rdev->endpoints)) { node = rdev->endpoints.next; ept = metal_container_of(node, struct rpmsg_endpoint, node); rpmsg_destroy_ept(ept); if (ept->ns_unbind_cb) { ept->ns_unbind_cb(ept); } } metal_mutex_deinit(&rdev->lock); rpmsg_port_destroy_queues(port); } /**************************************************************************** * Name: rpmsg_port_queue_get_available_buffer ****************************************************************************/ FAR struct rpmsg_port_header_s * rpmsg_port_queue_get_available_buffer(FAR struct rpmsg_port_queue_s *queue, bool wait) { FAR struct list_node *node; FAR struct rpmsg_port_header_s *hdr; for (; ; ) { node = rpmsg_port_remove_node(&queue->free); if (node) { hdr = RPMSG_PORT_NODE_TO_BUF(queue, node); hdr->len = queue->len; return hdr; } else if (!wait) { return NULL; } nxsem_wait_uninterruptible(&queue->free.sem); } } /**************************************************************************** * Name: rpmsg_port_queue_return_buffer ****************************************************************************/ FAR void rpmsg_port_queue_return_buffer(FAR struct rpmsg_port_queue_s *queue, FAR struct rpmsg_port_header_s *hdr) { FAR struct list_node *node = RPMSG_PORT_BUF_TO_NODE(queue, hdr); rpmsg_port_add_node(&queue->free, node); rpmsg_port_post(&queue->free.sem); } /**************************************************************************** * Name: rpmsg_port_queue_get_buffer ****************************************************************************/ FAR struct rpmsg_port_header_s * rpmsg_port_queue_get_buffer(FAR struct rpmsg_port_queue_s *queue, bool wait) { FAR struct list_node *node; for (; ; ) { node = rpmsg_port_remove_node(&queue->ready); if (node) { return RPMSG_PORT_NODE_TO_BUF(queue, node); } else if (!wait) { return NULL; } nxsem_wait_uninterruptible(&queue->ready.sem); } } /**************************************************************************** * Name: rpmsg_port_queue_add_buffer ****************************************************************************/ void rpmsg_port_queue_add_buffer(FAR struct rpmsg_port_queue_s *queue, FAR struct rpmsg_port_header_s *hdr) { FAR struct list_node *node = RPMSG_PORT_BUF_TO_NODE(queue, hdr); rpmsg_port_add_node(&queue->ready, node); rpmsg_port_post(&queue->ready.sem); } /**************************************************************************** * Name: rpmsg_port_register ****************************************************************************/ int rpmsg_port_register(FAR struct rpmsg_port_s *port, FAR const char *local_cpuname) { char name[64]; int ret; if (local_cpuname) { strlcpy(port->local_cpuname, local_cpuname, RPMSG_NAME_SIZE); } snprintf(name, sizeof(name), "/dev/rpmsg/%s", port->cpuname); ret = rpmsg_register(name, &port->rpmsg, &g_rpmsg_port_ops); if (ret < 0) { return ret; } rpmsg_device_created(&port->rpmsg); return ret; } /**************************************************************************** * Name: rpmsg_port_unregister ****************************************************************************/ void rpmsg_port_unregister(FAR struct rpmsg_port_s *port) { char name[64]; snprintf(name, sizeof(name), "/dev/rpmsg/%s", port->cpuname); rpmsg_unregister(name, &port->rpmsg); rpmsg_device_destory(&port->rpmsg); } /**************************************************************************** * Name: rpmsg_port_dump_buffer ****************************************************************************/ static void rpmsg_port_dump_buffer(FAR struct rpmsg_device *rdev, FAR struct rpmsg_port_queue_s *queue, bool rx) { FAR struct list_node *node; irqstate_t flags = spin_lock_irqsave(&queue->ready.lock); metal_log(METAL_LOG_EMERGENCY, "rpmsg_port queue %s: {used: %u, avail: %u}\n", rx ? "RX" : "TX", rpmsg_port_queue_nused(queue), rpmsg_port_queue_navail(queue)); metal_log(METAL_LOG_EMERGENCY, "rpmsg buffer list:\n"); list_for_every(&queue->ready.head, node) { FAR struct rpmsg_port_header_s *hdr = RPMSG_PORT_NODE_TO_BUF(queue, node); FAR struct rpmsg_hdr *rphdr = (FAR struct rpmsg_hdr *)hdr->buf; FAR struct rpmsg_endpoint *ept; ept = rpmsg_get_ept_from_addr(rdev, rx ? rphdr->dst : rphdr->src); if (ept) { metal_log(METAL_LOG_EMERGENCY, " %s buffer %p hold by %s\n", rx ? "RX" : "TX", rphdr, ept->name); } } spin_unlock_irqrestore(&queue->ready.lock, flags); } /**************************************************************************** * Name: rpmsg_port_dump ****************************************************************************/ static void rpmsg_port_dump(FAR struct rpmsg_s *rpmsg) { FAR struct rpmsg_port_s *port = (FAR struct rpmsg_port_s *)rpmsg; FAR struct rpmsg_device *rdev = rpmsg->rdev; FAR struct rpmsg_endpoint *ept; FAR struct metal_list *node; bool needunlock = false; if (!up_interrupt_context() && !sched_idletask() && !nxmutex_is_hold(&rdev->lock)) { metal_mutex_acquire(&rdev->lock); needunlock = true; } metal_log(METAL_LOG_EMERGENCY, "Remote: %s\n", port->cpuname); metal_list_for_each(&rdev->endpoints, node) { ept = metal_container_of(node, struct rpmsg_endpoint, node); metal_log(METAL_LOG_EMERGENCY, "ept %s\n", ept->name); } rpmsg_port_dump_buffer(rdev, &port->rxq, true); rpmsg_port_dump_buffer(rdev, &port->txq, false); if (needunlock) { metal_mutex_release(&rdev->lock); } }