incubator-nuttx/drivers/usrsock/usrsock_rpmsg_server.c

1321 lines
40 KiB
C

/****************************************************************************
* drivers/usrsock/usrsock_rpmsg_server.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 <debug.h>
#include <errno.h>
#include <inttypes.h>
#include <poll.h>
#include <string.h>
#include <sys/ioctl.h>
#include <nuttx/kmalloc.h>
#include <nuttx/mutex.h>
#include <nuttx/net/dns.h>
#include <nuttx/net/net.h>
#include <nuttx/queue.h>
#include <nuttx/rpmsg/rpmsg.h>
#include <nuttx/usrsock/usrsock_rpmsg.h>
#ifdef CONFIG_NETDEV_WIRELESS_IOCTL
#include <nuttx/wireless/wireless.h>
#endif
/****************************************************************************
* Private Types
****************************************************************************/
struct usrsock_rpmsg_s
{
rmutex_t mutex;
struct socket socks[CONFIG_NET_USRSOCK_RPMSG_SERVER_NSOCKS];
FAR struct rpmsg_endpoint *epts[CONFIG_NET_USRSOCK_RPMSG_SERVER_NSOCKS];
struct pollfd pfds[CONFIG_NET_USRSOCK_RPMSG_SERVER_NSOCKS];
};
/* Saving rpmsg requests to keep message order. */
struct usrsock_rpmsg_req_s
{
sq_entry_t flink;
FAR void *data;
size_t len;
uint32_t src;
};
struct usrsock_rpmsg_ept_s
{
struct rpmsg_endpoint ept;
/* For sendto/recvfrom */
struct iovec iov[CONFIG_NET_USRSOCK_RPMSG_SERVER_NIOVEC];
ssize_t remain;
/* For keeping msg order, normally nIOVec is the max request we can get */
bool inuse;
sq_queue_t req_free;
sq_queue_t req_pending;
struct usrsock_rpmsg_req_s reqs[CONFIG_NET_USRSOCK_RPMSG_SERVER_NIOVEC];
};
/****************************************************************************
* Private Function Prototypes
****************************************************************************/
static bool usrsock_rpmsg_available(FAR struct socket *psock, int cmd);
static int usrsock_rpmsg_send_ack(FAR struct rpmsg_endpoint *ept,
uint16_t events,
uint32_t xid, int32_t result);
static int usrsock_rpmsg_send_data_ack(FAR struct rpmsg_endpoint *ept,
FAR struct usrsock_message_datareq_ack_s *ack,
uint16_t events,
uint32_t xid, int32_t result,
uint16_t valuelen,
uint16_t valuelen_nontrunc,
int32_t datalen);
static int usrsock_rpmsg_send_event(FAR struct rpmsg_endpoint *ept,
int16_t usockid, uint16_t events);
static int usrsock_rpmsg_socket_handler(FAR struct rpmsg_endpoint *ept,
FAR void *data, size_t len,
uint32_t src, FAR void *priv_);
static int usrsock_rpmsg_close_handler(FAR struct rpmsg_endpoint *ept,
FAR void *data, size_t len,
uint32_t src, FAR void *priv_);
static int usrsock_rpmsg_connect_handler(FAR struct rpmsg_endpoint *ept,
FAR void *data, size_t len,
uint32_t src, FAR void *priv_);
static int usrsock_rpmsg_sendto_handler(FAR struct rpmsg_endpoint *ept,
FAR void *data, size_t len,
uint32_t src, FAR void *priv_);
static int usrsock_rpmsg_recvfrom_handler(FAR struct rpmsg_endpoint *ept,
FAR void *data, size_t len,
uint32_t src, FAR void *priv_);
static int usrsock_rpmsg_setsockopt_handler(FAR struct rpmsg_endpoint *ept,
FAR void *data, size_t len,
uint32_t src, FAR void *priv_);
static int usrsock_rpmsg_getsockopt_handler(FAR struct rpmsg_endpoint *ept,
FAR void *data, size_t len,
uint32_t src, FAR void *priv_);
static int usrsock_rpmsg_getsockname_handler(FAR struct rpmsg_endpoint *ept,
FAR void *data, size_t len,
uint32_t src, FAR void *priv_);
static int usrsock_rpmsg_getpeername_handler(FAR struct rpmsg_endpoint *ept,
FAR void *data, size_t len,
uint32_t src, FAR void *priv_);
static int usrsock_rpmsg_bind_handler(FAR struct rpmsg_endpoint *ept,
FAR void *data, size_t len,
uint32_t src, FAR void *priv_);
static int usrsock_rpmsg_listen_handler(FAR struct rpmsg_endpoint *ept,
FAR void *data, size_t len,
uint32_t src, FAR void *priv_);
static int usrsock_rpmsg_accept_handler(FAR struct rpmsg_endpoint *ept,
FAR void *data, size_t len,
uint32_t src, FAR void *priv_);
static int usrsock_rpmsg_ioctl_handler(FAR struct rpmsg_endpoint *ept,
FAR void *data, size_t len,
uint32_t src, FAR void *priv_);
static int usrsock_rpmsg_shutdown_handler(FAR struct rpmsg_endpoint *ept,
FAR void *data, size_t len,
uint32_t src, FAR void *priv_);
static int usrsock_rpmsg_dns_handler(FAR struct rpmsg_endpoint *ept,
FAR void *data, size_t len,
uint32_t src, FAR void *priv_);
static bool usrsock_rpmsg_ns_match(FAR struct rpmsg_device *rdev,
FAR void *priv_, FAR const char *name,
uint32_t dest);
static void usrsock_rpmsg_ns_bind(FAR struct rpmsg_device *rdev,
FAR void *priv_, FAR const char *name,
uint32_t dest);
static int usrsock_rpmsg_ept_cb(FAR struct rpmsg_endpoint *ept,
FAR void *data, size_t len, uint32_t src,
FAR void *priv);
static void usrsock_rpmsg_poll_cb(FAR struct pollfd *pfds);
static void usrsock_rpmsg_poll_setup(FAR struct pollfd *pfds,
pollevent_t events);
/****************************************************************************
* Private Data
****************************************************************************/
static const rpmsg_ept_cb g_usrsock_rpmsg_handler[] =
{
usrsock_rpmsg_socket_handler,
usrsock_rpmsg_close_handler,
usrsock_rpmsg_connect_handler,
usrsock_rpmsg_sendto_handler,
usrsock_rpmsg_recvfrom_handler,
usrsock_rpmsg_setsockopt_handler,
usrsock_rpmsg_getsockopt_handler,
usrsock_rpmsg_getsockname_handler,
usrsock_rpmsg_getpeername_handler,
usrsock_rpmsg_bind_handler,
usrsock_rpmsg_listen_handler,
usrsock_rpmsg_accept_handler,
usrsock_rpmsg_ioctl_handler,
usrsock_rpmsg_shutdown_handler,
usrsock_rpmsg_dns_handler,
};
/****************************************************************************
* Private Functions
****************************************************************************/
static bool usrsock_rpmsg_available(FAR struct socket *psock, int cmd)
{
int len;
if (psock_ioctl(psock, cmd, &len, sizeof(len)) == 0)
{
if (len > 0)
{
return true;
}
}
return false;
}
static int usrsock_rpmsg_send_ack(FAR struct rpmsg_endpoint *ept,
uint16_t events,
uint32_t xid, int32_t result)
{
struct usrsock_message_req_ack_s ack;
ack.head.msgid = USRSOCK_MESSAGE_RESPONSE_ACK;
ack.head.flags = (result == -EINPROGRESS);
ack.head.events = events;
ack.xid = xid;
ack.result = result == -EINPROGRESS ? 0 : result;
return rpmsg_send(ept, &ack, sizeof(ack));
}
static int usrsock_rpmsg_send_data_ack(FAR struct rpmsg_endpoint *ept,
FAR struct usrsock_message_datareq_ack_s *ack,
uint16_t events,
uint32_t xid, int32_t result,
uint16_t valuelen,
uint16_t valuelen_nontrunc,
int32_t datalen)
{
int ret;
ack->reqack.head.msgid = USRSOCK_MESSAGE_RESPONSE_DATA_ACK;
ack->reqack.head.flags = 0;
ack->reqack.head.events = events;
ack->reqack.xid = xid;
ack->reqack.result = result;
if (result < 0)
{
valuelen = 0;
valuelen_nontrunc = 0;
datalen = 0;
}
else if (valuelen > valuelen_nontrunc)
{
valuelen = valuelen_nontrunc;
}
ack->valuelen = valuelen;
ack->valuelen_nontrunc = valuelen_nontrunc;
ret = rpmsg_send_nocopy(ept, ack, sizeof(*ack) + valuelen + datalen);
if (ret < 0)
{
rpmsg_release_tx_buffer(ept, ack);
}
return ret;
}
static int usrsock_rpmsg_send_frag_ack(FAR struct rpmsg_endpoint *ept,
FAR struct usrsock_message_frag_ack_s *ack,
uint16_t events,
uint32_t xid, int32_t result,
uint32_t datalen)
{
ack->reqack.head.msgid = USRSOCK_RPMSG_FRAG_RESPONSE;
ack->reqack.head.flags = 0;
ack->reqack.head.events = events;
ack->reqack.xid = xid;
ack->reqack.result = result;
ack->datalen = datalen;
return rpmsg_send_nocopy(ept, ack, sizeof(*ack) + datalen);
}
static int usrsock_rpmsg_send_event(FAR struct rpmsg_endpoint *ept,
int16_t usockid, uint16_t events)
{
struct usrsock_message_socket_event_s event;
event.head.msgid = USRSOCK_MESSAGE_SOCKET_EVENT;
event.head.flags = USRSOCK_MESSAGE_FLAG_EVENT;
event.head.events = events;
event.usockid = usockid;
return rpmsg_send(ept, &event, sizeof(event));
}
static int usrsock_rpmsg_socket_handler(FAR struct rpmsg_endpoint *ept,
FAR void *data, size_t len,
uint32_t src, FAR void *priv_)
{
FAR struct usrsock_request_socket_s *req = data;
FAR struct usrsock_rpmsg_s *priv = priv_;
uint16_t events = 0;
int ret = -ENFILE;
int retr;
int i;
nxrmutex_lock(&priv->mutex);
for (i = 0; i < CONFIG_NET_USRSOCK_RPMSG_SERVER_NSOCKS; i++)
{
if (priv->epts[i] == NULL)
{
priv->epts[i] = ept;
nxrmutex_unlock(&priv->mutex);
ret = psock_socket(req->domain, req->type | SOCK_NONBLOCK,
req->protocol, &priv->socks[i]);
if (ret >= 0)
{
ret = i; /* Return index as the usockid */
if (req->type != SOCK_STREAM && req->type != SOCK_SEQPACKET)
{
events = USRSOCK_EVENT_SENDTO_READY;
}
}
else
{
priv->epts[i] = NULL;
}
break;
}
}
if (i == CONFIG_NET_USRSOCK_RPMSG_SERVER_NSOCKS)
{
nxrmutex_unlock(&priv->mutex);
}
retr = usrsock_rpmsg_send_ack(ept, events, req->head.xid, ret);
if (retr >= 0 && ret >= 0 &&
req->type != SOCK_STREAM && req->type != SOCK_SEQPACKET)
{
usrsock_rpmsg_poll_setup(&priv->pfds[ret], POLLIN);
}
return retr;
}
static int usrsock_rpmsg_close_handler(FAR struct rpmsg_endpoint *ept,
FAR void *data, size_t len,
uint32_t src, FAR void *priv_)
{
FAR struct usrsock_request_close_s *req = data;
FAR struct usrsock_rpmsg_s *priv = priv_;
int ret = -EBADF;
if (req->usockid >= 0 &&
req->usockid < CONFIG_NET_USRSOCK_RPMSG_SERVER_NSOCKS)
{
usrsock_rpmsg_poll_setup(&priv->pfds[req->usockid], 0);
/* It's safe to close sock here */
ret = psock_close(&priv->socks[req->usockid]);
nxrmutex_lock(&priv->mutex);
priv->epts[req->usockid] = NULL;
nxrmutex_unlock(&priv->mutex);
}
return usrsock_rpmsg_send_ack(ept, 0, req->head.xid, ret);
}
static int usrsock_rpmsg_connect_handler(FAR struct rpmsg_endpoint *ept,
FAR void *data, size_t len,
uint32_t src, FAR void *priv_)
{
FAR struct usrsock_request_connect_s *req = data;
FAR struct usrsock_rpmsg_s *priv = priv_;
bool inprogress = false;
int ret = -EBADF;
int retr;
if (req->usockid >= 0 &&
req->usockid < CONFIG_NET_USRSOCK_RPMSG_SERVER_NSOCKS)
{
ret = psock_connect(&priv->socks[req->usockid],
(FAR const struct sockaddr *)(req + 1), req->addrlen);
}
retr = usrsock_rpmsg_send_ack(ept, 0, req->head.xid, ret);
if (ret == -EINPROGRESS)
{
inprogress = true;
ret = 0;
}
if (retr >= 0 && ret >= 0)
{
pollevent_t events = POLLIN;
if (inprogress)
{
events |= POLLOUT;
}
usrsock_rpmsg_poll_setup(&priv->pfds[req->usockid], events);
if (!inprogress)
{
retr = usrsock_rpmsg_send_event(ept, req->usockid,
USRSOCK_EVENT_SENDTO_READY);
}
}
return retr;
}
static int usrsock_rpmsg_sendto_handler(FAR struct rpmsg_endpoint *ept,
FAR void *data, size_t len,
uint32_t src, FAR void *priv_)
{
FAR struct usrsock_request_sendto_s *req;
FAR struct usrsock_rpmsg_ept_s *uept =
(FAR struct usrsock_rpmsg_ept_s *)ept;
FAR struct usrsock_rpmsg_s *priv = priv_;
uint16_t events = 0;
ssize_t ret = -EBADF;
int retr;
int i;
if (uept->remain > 0)
{
size_t hlen;
struct msghdr msg =
{
};
uept->remain -= len;
if (!uept->iov[0].iov_base)
{
/* Maybe error occurred previously, skip processing. */
return 0;
}
req = uept->iov[0].iov_base;
hlen = sizeof(*req) + req->addrlen;
for (i = 0; i < CONFIG_NET_USRSOCK_RPMSG_SERVER_NIOVEC; i++)
{
if (!uept->iov[i].iov_base)
{
uept->iov[i].iov_base = data;
uept->iov[i].iov_len = len;
rpmsg_hold_rx_buffer(ept, data);
break;
}
}
/* Partial packet ? continue to fetch */
if (uept->remain > 0)
{
/* We've used the last I/O vector, cannot continue. */
if (i == CONFIG_NET_USRSOCK_RPMSG_SERVER_NIOVEC - 1)
{
nerr("ERROR: Request %d too large!\n", req->usockid);
ret = -ENOMEM;
goto out;
}
return 0;
}
else if (uept->remain < 0)
{
ret = -EINVAL;
goto out;
}
/* Skip the sendto header from I/O vector */
uept->iov[0].iov_base = (FAR char *)uept->iov[0].iov_base + hlen;
uept->iov[0].iov_len -= hlen;
msg.msg_name = req->addrlen ? (FAR void *)(req + 1) : NULL;
msg.msg_namelen = req->addrlen;
msg.msg_iov = uept->iov;
msg.msg_iovlen = i + 1;
ret = psock_sendmsg(&priv->socks[req->usockid], &msg, req->flags);
/* Recover the I/O vector */
uept->iov[0].iov_base = (FAR char *)uept->iov[0].iov_base - hlen;
uept->iov[0].iov_len += hlen;
}
else
{
req = data;
if (req->usockid >= 0 &&
req->usockid < CONFIG_NET_USRSOCK_RPMSG_SERVER_NSOCKS)
{
uept->remain = sizeof(*req) + req->addrlen + req->buflen - len;
if (uept->remain > 0)
{
#if CONFIG_NET_USRSOCK_RPMSG_SERVER_NIOVEC >= 2
uept->iov[0].iov_base = data;
uept->iov[0].iov_len = len;
rpmsg_hold_rx_buffer(ept, data);
return 0;
#else
ret = -ENOMEM;
#endif
}
else
{
ret = psock_sendto(&priv->socks[req->usockid],
(FAR const void *)(req + 1) + req->addrlen, req->buflen,
req->flags,
req->addrlen ?
(FAR const struct sockaddr *)(req + 1) : NULL,
req->addrlen);
}
}
}
out:
if (ret > 0 &&
usrsock_rpmsg_available(&priv->socks[req->usockid], FIONSPACE))
{
events |= USRSOCK_EVENT_SENDTO_READY;
}
retr = usrsock_rpmsg_send_ack(ept, events, req->head.xid, ret);
if (retr >= 0 && (ret > 0 || ret == -EAGAIN) && events == 0)
{
usrsock_rpmsg_poll_setup(&priv->pfds[req->usockid],
priv->pfds[req->usockid].events | POLLOUT);
}
if (uept->iov[0].iov_base)
{
for (i = 0; i < CONFIG_NET_USRSOCK_RPMSG_SERVER_NIOVEC; i++)
{
if (uept->iov[i].iov_base == NULL)
{
break;
}
rpmsg_release_rx_buffer(ept, uept->iov[i].iov_base);
uept->iov[i].iov_base = NULL;
uept->iov[i].iov_len = 0;
}
}
return retr;
}
static int usrsock_rpmsg_recvfrom_handler(FAR struct rpmsg_endpoint *ept,
FAR void *data, size_t len_,
uint32_t src, FAR void *priv_)
{
struct iovec iov[CONFIG_NET_USRSOCK_RPMSG_SERVER_NIOVEC];
FAR struct usrsock_request_recvfrom_s *req = data;
FAR struct usrsock_message_datareq_ack_s *ack;
FAR struct usrsock_rpmsg_s *priv = priv_;
socklen_t outaddrlen = req->max_addrlen;
socklen_t inaddrlen = req->max_addrlen;
size_t buflen = req->max_buflen;
size_t totlen = 0;
ssize_t ret = -EBADF;
uint32_t len = buflen;
uint16_t events = 0;
uint8_t i = 0;
int retr;
ack = rpmsg_get_tx_payload_buffer(ept, &len, true);
if (sizeof(*ack) + inaddrlen + buflen < len)
{
len = sizeof(*ack) + inaddrlen + buflen;
}
memset(iov, 0, sizeof(iov));
if (req->usockid >= 0 &&
req->usockid < CONFIG_NET_USRSOCK_RPMSG_SERVER_NSOCKS)
{
ret = psock_recvfrom(&priv->socks[req->usockid],
(FAR void *)(ack + 1) + inaddrlen,
len - sizeof(*ack) - inaddrlen,
req->flags,
outaddrlen ? (FAR struct sockaddr *)(ack + 1) : NULL,
outaddrlen ? &outaddrlen : NULL);
totlen = ret;
if (ret > 0 && (priv->socks[req->usockid].s_type & SOCK_TYPE_MASK) ==
SOCK_STREAM)
{
if (outaddrlen < inaddrlen)
{
memcpy((FAR void *)(ack + 1) + outaddrlen,
(FAR void *)(ack + 1) + inaddrlen, ret);
}
/* Hold net_lock to combine get_tx_payload and recvfrom together.
* Otherwise we may keep holding tx buffer when waiting net_lock in
* recvfrom, which may block rpmsg and may cause dead lock if
* another thread tries to get tx buffer with net_lock held.
*/
net_lock();
while (totlen < buflen &&
i < CONFIG_NET_USRSOCK_RPMSG_SERVER_NIOVEC)
{
uint32_t offset = sizeof(struct usrsock_message_frag_ack_s);
if (!usrsock_rpmsg_available(&priv->socks[req->usockid],
FIONREAD))
{
break;
}
iov[i].iov_base = rpmsg_get_tx_payload_buffer(ept,
&len,
false);
if (!iov[i].iov_base)
{
break;
}
DEBUGASSERT(len > offset);
if (buflen - totlen < len - offset)
{
len = buflen - totlen + offset;
}
/* Should never wait */
iov[i].iov_len =
psock_recvfrom(&priv->socks[req->usockid],
(FAR char *)iov[i].iov_base + offset,
len - offset,
req->flags | MSG_DONTWAIT,
NULL, NULL);
if ((ssize_t)iov[i].iov_len > 0)
{
totlen += iov[i].iov_len;
if (iov[i].iov_len < len - offset)
{
break;
}
}
else
{
iov[i].iov_len = 0;
break;
}
i++;
}
if (usrsock_rpmsg_available(&priv->socks[req->usockid], FIONREAD))
{
events |= USRSOCK_EVENT_RECVFROM_AVAIL;
}
net_unlock();
}
}
retr = usrsock_rpmsg_send_data_ack(ept,
ack, events, req->head.xid,
totlen, inaddrlen, outaddrlen,
ret);
for (i = 0; i < CONFIG_NET_USRSOCK_RPMSG_SERVER_NIOVEC; i++)
{
if (!iov[i].iov_base)
{
break;
}
if (!iov[i].iov_len || retr <= 0)
{
rpmsg_release_tx_buffer(ept, iov[i].iov_base);
iov[i].iov_base = NULL;
continue;
}
usrsock_rpmsg_send_frag_ack(ept,
(FAR struct usrsock_message_frag_ack_s *)iov[i].iov_base,
events, req->head.xid, totlen, iov[i].iov_len);
}
if (retr >= 0 && (ret > 0 || ret == -EAGAIN) && events == 0)
{
usrsock_rpmsg_poll_setup(&priv->pfds[req->usockid],
priv->pfds[req->usockid].events | POLLIN);
}
return retr;
}
static int usrsock_rpmsg_setsockopt_handler(FAR struct rpmsg_endpoint *ept,
FAR void *data, size_t len,
uint32_t src, FAR void *priv_)
{
FAR struct usrsock_request_setsockopt_s *req = data;
FAR struct usrsock_rpmsg_s *priv = priv_;
int ret = -EBADF;
if (req->usockid >= 0 &&
req->usockid < CONFIG_NET_USRSOCK_RPMSG_SERVER_NSOCKS)
{
ret = psock_setsockopt(&priv->socks[req->usockid],
req->level, req->option, req + 1, req->valuelen);
}
return usrsock_rpmsg_send_ack(ept, 0, req->head.xid, ret);
}
static int usrsock_rpmsg_getsockopt_handler(FAR struct rpmsg_endpoint *ept,
FAR void *data, size_t len_,
uint32_t src, FAR void *priv_)
{
FAR struct usrsock_request_getsockopt_s *req = data;
FAR struct usrsock_message_datareq_ack_s *ack;
FAR struct usrsock_rpmsg_s *priv = priv_;
socklen_t optlen = req->max_valuelen;
int ret = -EBADF;
uint32_t len;
ack = rpmsg_get_tx_payload_buffer(ept, &len, true);
if (req->usockid >= 0 &&
req->usockid < CONFIG_NET_USRSOCK_RPMSG_SERVER_NSOCKS)
{
ret = psock_getsockopt(&priv->socks[req->usockid],
req->level, req->option, ack + 1, &optlen);
}
return usrsock_rpmsg_send_data_ack(ept,
ack, 0, req->head.xid, ret, optlen, optlen, ret);
}
static int usrsock_rpmsg_getsockname_handler(FAR struct rpmsg_endpoint *ept,
FAR void *data, size_t len_,
uint32_t src, FAR void *priv_)
{
FAR struct usrsock_request_getsockname_s *req = data;
FAR struct usrsock_message_datareq_ack_s *ack;
FAR struct usrsock_rpmsg_s *priv = priv_;
socklen_t outaddrlen = req->max_addrlen;
socklen_t inaddrlen = req->max_addrlen;
int ret = -EBADF;
uint32_t len;
ack = rpmsg_get_tx_payload_buffer(ept, &len, true);
if (req->usockid >= 0 &&
req->usockid < CONFIG_NET_USRSOCK_RPMSG_SERVER_NSOCKS)
{
ret = psock_getsockname(&priv->socks[req->usockid],
(FAR struct sockaddr *)(ack + 1), &outaddrlen);
}
return usrsock_rpmsg_send_data_ack(ept,
ack, 0, req->head.xid, ret, inaddrlen, outaddrlen, ret);
}
static int usrsock_rpmsg_getpeername_handler(FAR struct rpmsg_endpoint *ept,
FAR void *data, size_t len_,
uint32_t src, FAR void *priv_)
{
FAR struct usrsock_request_getpeername_s *req = data;
FAR struct usrsock_message_datareq_ack_s *ack;
FAR struct usrsock_rpmsg_s *priv = priv_;
socklen_t outaddrlen = req->max_addrlen;
socklen_t inaddrlen = req->max_addrlen;
int ret = -EBADF;
uint32_t len;
ack = rpmsg_get_tx_payload_buffer(ept, &len, true);
if (req->usockid >= 0 &&
req->usockid < CONFIG_NET_USRSOCK_RPMSG_SERVER_NSOCKS)
{
ret = psock_getpeername(&priv->socks[req->usockid],
(FAR struct sockaddr *)(ack + 1), &outaddrlen);
}
return usrsock_rpmsg_send_data_ack(ept,
ack, 0, req->head.xid, ret, inaddrlen, outaddrlen, ret);
}
static int usrsock_rpmsg_bind_handler(FAR struct rpmsg_endpoint *ept,
FAR void *data, size_t len,
uint32_t src, FAR void *priv_)
{
FAR struct usrsock_request_bind_s *req = data;
FAR struct usrsock_rpmsg_s *priv = priv_;
int ret = -EBADF;
if (req->usockid >= 0 &&
req->usockid < CONFIG_NET_USRSOCK_RPMSG_SERVER_NSOCKS)
{
ret = psock_bind(&priv->socks[req->usockid],
(FAR const struct sockaddr *)(req + 1), req->addrlen);
}
return usrsock_rpmsg_send_ack(ept, 0, req->head.xid, ret);
}
static int usrsock_rpmsg_listen_handler(FAR struct rpmsg_endpoint *ept,
FAR void *data, size_t len,
uint32_t src, FAR void *priv_)
{
FAR struct usrsock_request_listen_s *req = data;
FAR struct usrsock_rpmsg_s *priv = priv_;
int ret = -EBADF;
int retr;
if (req->usockid >= 0 &&
req->usockid < CONFIG_NET_USRSOCK_RPMSG_SERVER_NSOCKS)
{
ret = psock_listen(&priv->socks[req->usockid], req->backlog);
}
retr = usrsock_rpmsg_send_ack(ept, 0, req->head.xid, ret);
if (retr >= 0 && ret >= 0)
{
usrsock_rpmsg_poll_setup(&priv->pfds[req->usockid], POLLIN);
}
return retr;
}
static int usrsock_rpmsg_accept_handler(FAR struct rpmsg_endpoint *ept,
FAR void *data, size_t len_,
uint32_t src, FAR void *priv_)
{
FAR struct usrsock_request_accept_s *req = data;
FAR struct usrsock_message_datareq_ack_s *ack;
FAR struct usrsock_rpmsg_s *priv = priv_;
socklen_t outaddrlen = req->max_addrlen;
socklen_t inaddrlen = req->max_addrlen;
uint32_t len;
int ret = -EBADF;
int i = 0;
int retr;
ack = rpmsg_get_tx_payload_buffer(ept, &len, true);
if (req->usockid >= 0 &&
req->usockid < CONFIG_NET_USRSOCK_RPMSG_SERVER_NSOCKS)
{
ret = -ENFILE; /* Assume no free socket handler */
nxrmutex_lock(&priv->mutex);
for (i = 0; i < CONFIG_NET_USRSOCK_RPMSG_SERVER_NSOCKS; i++)
{
if (priv->epts[i] == NULL)
{
priv->epts[i] = ept;
nxrmutex_unlock(&priv->mutex);
ret = psock_accept(&priv->socks[req->usockid],
outaddrlen ? (FAR struct sockaddr *)(ack + 1) : NULL,
outaddrlen ? &outaddrlen : NULL, &priv->socks[i],
SOCK_NONBLOCK);
if (ret >= 0)
{
/* Append index as usockid to the payload */
if (outaddrlen <= inaddrlen)
{
*(FAR int16_t *)
((FAR void *)(ack + 1) + outaddrlen) = i;
}
else
{
*(FAR int16_t *)
((FAR void *)(ack + 1) + inaddrlen) = i;
}
ret = sizeof(int16_t); /* Return usockid size */
}
else
{
priv->epts[i] = NULL;
}
break;
}
}
if (i == CONFIG_NET_USRSOCK_RPMSG_SERVER_NSOCKS)
{
nxrmutex_unlock(&priv->mutex);
}
}
retr = usrsock_rpmsg_send_data_ack(ept,
ack, 0, req->head.xid, ret, inaddrlen, outaddrlen, ret);
if (retr >= 0 && ret >= 0)
{
usrsock_rpmsg_poll_setup(&priv->pfds[req->usockid], POLLIN);
usrsock_rpmsg_poll_setup(&priv->pfds[i], POLLIN);
usrsock_rpmsg_send_event(ept, i, USRSOCK_EVENT_SENDTO_READY);
}
return retr;
}
static int usrsock_rpmsg_ioctl_handler(FAR struct rpmsg_endpoint *ept,
FAR void *data, size_t len_,
uint32_t src, FAR void *priv_)
{
FAR struct usrsock_request_ioctl_s *req = data;
FAR struct usrsock_message_datareq_ack_s *ack;
FAR struct usrsock_rpmsg_s *priv = priv_;
#ifdef CONFIG_NETDEV_WIRELESS_IOCTL
FAR struct iwreq *wlreq;
FAR struct iwreq *wlack;
#endif
int ret = -EBADF;
uint32_t len;
ack = rpmsg_get_tx_payload_buffer(ept, &len, true);
if (req->usockid >= 0 &&
req->usockid < CONFIG_NET_USRSOCK_RPMSG_SERVER_NSOCKS)
{
memcpy(ack + 1, req + 1, len_ - sizeof(*req));
#ifdef CONFIG_NETDEV_WIRELESS_IOCTL
wlreq = (FAR struct iwreq *)(req + 1);
wlack = (FAR struct iwreq *)(ack + 1);
if (WL_IS80211POINTERCMD(req->cmd) && wlreq->u.data.pointer)
{
wlack->u.data.pointer = wlack + 1;
}
#endif
ret = psock_ioctl(&priv->socks[req->usockid],
req->cmd, (unsigned long)(ack + 1));
#ifdef CONFIG_NETDEV_WIRELESS_IOCTL
if (WL_IS80211POINTERCMD(req->cmd) && wlreq->u.data.pointer)
{
if (ret >= 0)
{
ret = wlreq->u.data.length;
}
wlack->u.data.pointer = wlreq->u.data.pointer;
}
#endif
}
return usrsock_rpmsg_send_data_ack(ept,
ack, 0, req->head.xid, ret, req->arglen, req->arglen, ret);
}
static int usrsock_rpmsg_shutdown_handler(FAR struct rpmsg_endpoint *ept,
FAR void *data, size_t len,
uint32_t src, FAR void *priv_)
{
FAR struct usrsock_request_shutdown_s *req = data;
FAR struct usrsock_rpmsg_s *priv = priv_;
int ret = -EBADF;
if (req->usockid >= 0 &&
req->usockid < CONFIG_NET_USRSOCK_RPMSG_SERVER_NSOCKS)
{
ret = psock_shutdown(&priv->socks[req->usockid], req->how);
}
return usrsock_rpmsg_send_ack(ept, 0, req->head.xid, ret);
}
static int usrsock_rpmsg_dns_handler(FAR struct rpmsg_endpoint *ept,
FAR void *data, size_t len,
uint32_t src, FAR void *priv_)
{
#ifdef CONFIG_NETDB_DNSCLIENT
FAR struct usrsock_rpmsg_dns_request_s *dns = data;
dns_add_nameserver((FAR struct sockaddr *)(dns + 1), dns->addrlen);
#endif
return 0;
}
#ifdef CONFIG_NETDB_DNSCLIENT
static int usrsock_rpmsg_send_dns_event(FAR void *arg,
FAR struct sockaddr *addr,
socklen_t addrlen)
{
FAR struct rpmsg_endpoint *ept = arg;
FAR struct usrsock_rpmsg_dns_event_s *dns;
uint32_t len;
int ret;
dns = rpmsg_get_tx_payload_buffer(ept, &len, true);
if (dns == NULL)
{
return -ENOMEM;
}
dns->head.msgid = USRSOCK_RPMSG_DNS_EVENT;
dns->head.flags = USRSOCK_MESSAGE_FLAG_EVENT;
dns->addrlen = addrlen;
memcpy(dns + 1, addr, addrlen);
ret = rpmsg_send_nocopy(ept, dns, sizeof(*dns) + addrlen);
if (ret < 0)
{
rpmsg_release_tx_buffer(ept, dns);
}
return ret;
}
#endif
static bool usrsock_rpmsg_ns_match(FAR struct rpmsg_device *rdev,
FAR void *priv_, FAR const char *name,
uint32_t dest)
{
return !strcmp(name, USRSOCK_RPMSG_EPT_NAME);
}
static void usrsock_rpmsg_ept_release(FAR struct rpmsg_endpoint *ept)
{
FAR struct usrsock_rpmsg_s *priv = ept->priv;
int i;
#ifdef CONFIG_NETDB_DNSCLIENT
dns_unregister_notify(usrsock_rpmsg_send_dns_event, ept);
#endif
/* Collect all socks belong to the dead client */
for (i = 0; i < CONFIG_NET_USRSOCK_RPMSG_SERVER_NSOCKS; i++)
{
if (priv->epts[i] == ept)
{
usrsock_rpmsg_poll_setup(&priv->pfds[i], 0);
/* It's safe to close socks here */
psock_close(&priv->socks[i]);
nxrmutex_lock(&priv->mutex);
priv->epts[i] = NULL;
nxrmutex_unlock(&priv->mutex);
}
}
kmm_free(ept);
}
static void usrsock_rpmsg_ns_bind(FAR struct rpmsg_device *rdev,
FAR void *priv_, FAR const char *name,
uint32_t dest)
{
FAR struct usrsock_rpmsg_s *priv = priv_;
FAR struct usrsock_rpmsg_ept_s *uept;
int ret;
int i;
uept = kmm_zalloc(sizeof(*uept));
if (!uept)
{
return;
}
uept->ept.priv = priv;
uept->ept.release_cb = usrsock_rpmsg_ept_release;
for (i = 0; i < CONFIG_NET_USRSOCK_RPMSG_SERVER_NIOVEC; i++)
{
sq_addlast(&uept->reqs[i].flink, &uept->req_free);
}
ret = rpmsg_create_ept(&uept->ept, rdev, USRSOCK_RPMSG_EPT_NAME,
RPMSG_ADDR_ANY, dest,
usrsock_rpmsg_ept_cb, rpmsg_destroy_ept);
if (ret < 0)
{
kmm_free(uept);
return;
}
#ifdef CONFIG_NETDB_DNSCLIENT
dns_register_notify(usrsock_rpmsg_send_dns_event, &uept->ept);
#endif
}
static int usrsock_rpmsg_ept_do_cb(FAR struct usrsock_rpmsg_ept_s *uept,
FAR void *data, size_t len, uint32_t src,
FAR struct usrsock_rpmsg_s *priv)
{
FAR struct usrsock_request_common_s *common = data;
if (uept->remain > 0)
{
return usrsock_rpmsg_sendto_handler(&uept->ept, data, len, src, priv);
}
else if (common->reqid >= 0 && common->reqid <= USRSOCK_REQUEST__MAX)
{
return g_usrsock_rpmsg_handler[common->reqid](&uept->ept, data, len,
src, priv);
}
return -EINVAL;
}
static int usrsock_rpmsg_ept_cb(FAR struct rpmsg_endpoint *ept,
FAR void *data, size_t len, uint32_t src,
FAR void *priv_)
{
FAR struct usrsock_rpmsg_s *priv = priv_;
FAR struct usrsock_rpmsg_ept_s *uept =
(FAR struct usrsock_rpmsg_ept_s *)ept;
FAR struct usrsock_rpmsg_req_s *req;
int ret;
/* This callback is called from only one thread per ept, so don't need any
* lock for `inuse` state.
*/
if (uept->inuse)
{
/* Avoid recursive call. */
req = (FAR struct usrsock_rpmsg_req_s *)sq_remfirst(&uept->req_free);
if (req == NULL)
{
return -ENOMEM;
}
req->data = data;
req->len = len;
req->src = src;
sq_addlast(&req->flink, &uept->req_pending);
rpmsg_hold_rx_buffer(ept, data);
return OK;
}
uept->inuse = true;
ret = usrsock_rpmsg_ept_do_cb(uept, data, len, src, priv);
/* Pop pending requests to proceed. */
while ((req = (FAR struct usrsock_rpmsg_req_s *)
sq_remfirst(&uept->req_pending)) != NULL)
{
data = req->data;
len = req->len;
src = req->src;
sq_addlast(&req->flink, &uept->req_free);
ret = usrsock_rpmsg_ept_do_cb(uept, data, len, src, priv);
if (ret < 0)
{
nerr("ERROR: usrsock got error %d!", ret);
}
rpmsg_release_rx_buffer(ept, data);
}
uept->inuse = false;
return ret;
}
static void usrsock_rpmsg_poll_setup(FAR struct pollfd *pfds,
pollevent_t events)
{
FAR struct usrsock_rpmsg_s *priv = (FAR struct usrsock_rpmsg_s *)pfds->arg;
FAR struct socket *psock = &priv->socks[pfds->fd];
int ret = 0;
/* No poll for SOCK_CTRL. */
if (psock->s_type == SOCK_CTRL)
{
return;
}
net_lock();
if (events)
{
if (pfds->events)
{
ret = psock_poll(psock, pfds, false);
}
if (ret >= 0)
{
/* The protocol stack monitor flag is different when the events is
* POLLIN or POLLOUT, so we have to call poll_setup again.
*/
pfds->revents = 0;
pfds->events = events;
ret = psock_poll(psock, pfds, true);
}
}
else
{
pfds->revents = 0;
pfds->events = 0;
ret = psock_poll(psock, pfds, false);
}
if (ret < 0)
{
nerr("psock_poll failed. ret %d domain %u type %u pfds->fd %d"
", pfds->events %08" PRIx32 ", pfds->revents %08" PRIx32,
ret, psock->s_domain, psock->s_type, pfds->fd,
pfds->events, pfds->revents);
}
net_unlock();
}
static void usrsock_rpmsg_poll_cb(FAR struct pollfd *pfds)
{
FAR struct usrsock_rpmsg_s *priv = (FAR struct usrsock_rpmsg_s *)pfds->arg;
int oldevents;
int events = 0;
nxrmutex_lock(&priv->mutex);
if (!priv->epts[pfds->fd])
{
nxrmutex_unlock(&priv->mutex);
return;
}
oldevents = pfds->events;
if (pfds->revents & POLLIN)
{
events |= USRSOCK_EVENT_RECVFROM_AVAIL;
/* Stop poll in until recv get called */
pfds->events &= ~POLLIN;
pfds->revents &= ~POLLIN;
}
if (pfds->revents & POLLOUT)
{
events |= USRSOCK_EVENT_SENDTO_READY;
/* Stop poll out until send get called */
pfds->events &= ~POLLOUT;
pfds->revents &= ~POLLOUT;
}
if (pfds->revents & (POLLHUP | POLLERR))
{
events |= USRSOCK_EVENT_REMOTE_CLOSED;
/* Check data that has not been recv */
if (usrsock_rpmsg_available(&priv->socks[pfds->fd], FIONREAD))
{
events |= USRSOCK_EVENT_RECVFROM_AVAIL;
}
/* Clear revents */
pfds->revents &= ~(POLLHUP | POLLERR);
}
if (oldevents != pfds->events)
{
usrsock_rpmsg_poll_setup(pfds, pfds->events);
}
if (events != 0)
{
usrsock_rpmsg_send_event(priv->epts[pfds->fd], pfds->fd, events);
}
nxrmutex_unlock(&priv->mutex);
}
/****************************************************************************
* Public Functions
****************************************************************************/
int usrsock_rpmsg_server_initialize(void)
{
FAR struct usrsock_rpmsg_s *priv;
int ret;
int i;
priv = kmm_calloc(1, sizeof(*priv));
if (priv == NULL)
{
return -ENOMEM;
}
nxrmutex_init(&priv->mutex);
/* Setup poll callback settings */
for (i = 0; i < CONFIG_NET_USRSOCK_RPMSG_SERVER_NSOCKS; i++)
{
priv->pfds[i].fd = i;
priv->pfds[i].arg = priv;
priv->pfds[i].cb = usrsock_rpmsg_poll_cb;
}
ret = rpmsg_register_callback(priv,
NULL,
NULL,
usrsock_rpmsg_ns_match,
usrsock_rpmsg_ns_bind);
if (ret >= 0)
{
return ret;
}
nxrmutex_destroy(&priv->mutex);
kmm_free(priv);
return ret;
}