6a3c2aded6
* Simplify EINTR/ECANCEL error handling 1. Add semaphore uninterruptible wait function 2 .Replace semaphore wait loop with a single uninterruptible wait 3. Replace all sem_xxx to nxsem_xxx * Unify the void cast usage 1. Remove void cast for function because many place ignore the returned value witout cast 2. Replace void cast for variable with UNUSED macro
350 lines
10 KiB
C
350 lines
10 KiB
C
/****************************************************************************
|
|
* net/local/local_connect.c
|
|
*
|
|
* Copyright (C) 2015-2017 Gregory Nutt. All rights reserved.
|
|
* Author: Gregory Nutt <gnutt@nuttx.org>
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
*
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in
|
|
* the documentation and/or other materials provided with the
|
|
* distribution.
|
|
* 3. Neither the name NuttX nor the names of its contributors may be
|
|
* used to endorse or promote products derived from this software
|
|
* without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
|
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
|
|
* COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
|
|
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
|
|
* OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
|
|
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
|
|
* ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
*
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
* Included Files
|
|
****************************************************************************/
|
|
|
|
#include <nuttx/config.h>
|
|
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
#include <assert.h>
|
|
#include <errno.h>
|
|
#include <queue.h>
|
|
#include <debug.h>
|
|
|
|
#include <nuttx/semaphore.h>
|
|
#include <nuttx/net/net.h>
|
|
|
|
#include <arch/irq.h>
|
|
|
|
#include "utils/utils.h"
|
|
#include "socket/socket.h"
|
|
#include "local/local.h"
|
|
|
|
#ifdef CONFIG_NET_LOCAL_STREAM
|
|
|
|
/****************************************************************************
|
|
* Private Functions
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
* Name: local_generate_instance_id
|
|
****************************************************************************/
|
|
|
|
static int32_t local_generate_instance_id(void)
|
|
{
|
|
static int32_t g_next_instance_id = 0;
|
|
int32_t id;
|
|
|
|
/* Called from local_connect with net_lock held. */
|
|
|
|
id = g_next_instance_id++;
|
|
if (g_next_instance_id < 0)
|
|
{
|
|
g_next_instance_id = 0;
|
|
}
|
|
|
|
return id;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: _local_semtake() and _local_semgive()
|
|
*
|
|
* Description:
|
|
* Take/give semaphore
|
|
*
|
|
****************************************************************************/
|
|
|
|
static inline void _local_semtake(sem_t *sem)
|
|
{
|
|
net_lockedwait_uninterruptible(sem);
|
|
}
|
|
|
|
#define _local_semgive(sem) nxsem_post(sem)
|
|
|
|
/****************************************************************************
|
|
* Name: local_stream_connect
|
|
*
|
|
* Description:
|
|
* Find a local connection structure that is the appropriate "server"
|
|
* connection to be used with the provided "client" connection.
|
|
*
|
|
* Returned Value:
|
|
* Zero (OK) returned on success; A negated errno value is returned on a
|
|
* failure. Possible failures include:
|
|
*
|
|
* Assumptions:
|
|
* The network is locked on entry, unlocked on return. This logic is
|
|
* an integral part of the lock_connect() implementation and was
|
|
* separated out only to improve readability.
|
|
*
|
|
****************************************************************************/
|
|
|
|
static int inline local_stream_connect(FAR struct local_conn_s *client,
|
|
FAR struct local_conn_s *server,
|
|
bool nonblock)
|
|
{
|
|
int ret;
|
|
int sval;
|
|
|
|
/* Has server backlog been reached?
|
|
* NOTE: The backlog will be zero if listen() has never been called by the
|
|
* server.
|
|
*/
|
|
|
|
if (server->lc_state != LOCAL_STATE_LISTENING ||
|
|
server->u.server.lc_pending >= server->u.server.lc_backlog)
|
|
{
|
|
net_unlock();
|
|
nerr("ERROR: Server is not listening: lc_state=%d\n",
|
|
server->lc_state);
|
|
nerr(" OR: The backlog limit was reached: %d or %d\n",
|
|
server->u.server.lc_pending, server->u.server.lc_backlog);
|
|
return -ECONNREFUSED;
|
|
}
|
|
|
|
/* Increment the number of pending server connection s */
|
|
|
|
server->u.server.lc_pending++;
|
|
DEBUGASSERT(server->u.server.lc_pending != 0);
|
|
|
|
/* Create the FIFOs needed for the connection */
|
|
|
|
ret = local_create_fifos(client);
|
|
if (ret < 0)
|
|
{
|
|
nerr("ERROR: Failed to create FIFOs for %s: %d\n",
|
|
client->lc_path, ret);
|
|
|
|
net_unlock();
|
|
return ret;
|
|
}
|
|
|
|
/* Open the client-side write-only FIFO. This should not block and should
|
|
* prevent the server-side from blocking as well.
|
|
*/
|
|
|
|
ret = local_open_client_tx(client, nonblock);
|
|
if (ret < 0)
|
|
{
|
|
nerr("ERROR: Failed to open write-only FIFOs for %s: %d\n",
|
|
client->lc_path, ret);
|
|
|
|
net_unlock();
|
|
goto errout_with_fifos;
|
|
}
|
|
|
|
DEBUGASSERT(client->lc_outfile.f_inode != NULL);
|
|
|
|
/* Set the busy "result" before giving the semaphore. */
|
|
|
|
client->u.client.lc_result = -EBUSY;
|
|
|
|
/* Add ourself to the list of waiting connections and notify the server. */
|
|
|
|
dq_addlast(&client->lc_node, &server->u.server.lc_waiters);
|
|
client->lc_state = LOCAL_STATE_ACCEPT;
|
|
local_accept_pollnotify(server, POLLIN);
|
|
|
|
if (nxsem_getvalue(&server->lc_waitsem, &sval) >= 0 && sval < 1)
|
|
{
|
|
_local_semgive(&server->lc_waitsem);
|
|
}
|
|
|
|
net_unlock();
|
|
|
|
/* Wait for the server to accept the connections */
|
|
|
|
do
|
|
{
|
|
_local_semtake(&client->lc_waitsem);
|
|
ret = client->u.client.lc_result;
|
|
}
|
|
while (ret == -EBUSY);
|
|
|
|
/* Did we successfully connect? */
|
|
|
|
if (ret < 0)
|
|
{
|
|
nerr("ERROR: Failed to connect: %d\n", ret);
|
|
goto errout_with_outfd;
|
|
}
|
|
|
|
/* Yes.. open the read-only FIFO */
|
|
|
|
ret = local_open_client_rx(client, nonblock);
|
|
if (ret < 0)
|
|
{
|
|
nerr("ERROR: Failed to open write-only FIFOs for %s: %d\n",
|
|
client->lc_path, ret);
|
|
goto errout_with_outfd;
|
|
}
|
|
|
|
DEBUGASSERT(client->lc_infile.f_inode != NULL);
|
|
client->lc_state = LOCAL_STATE_CONNECTED;
|
|
return OK;
|
|
|
|
errout_with_outfd:
|
|
file_close(&client->lc_outfile);
|
|
client->lc_outfile.f_inode = NULL;
|
|
|
|
errout_with_fifos:
|
|
local_release_fifos(client);
|
|
client->lc_state = LOCAL_STATE_BOUND;
|
|
return ret;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Public Functions
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
* Name: psock_local_connect
|
|
*
|
|
* Description:
|
|
* Find a local connection structure that is the appropriate "server"
|
|
* connection to be used with the provided "client" connection.
|
|
*
|
|
* Returned Value:
|
|
* Zero (OK) returned on success; A negated errno value is returned on a
|
|
* failure. Possible failures include:
|
|
*
|
|
* EISCONN - The specified socket is connection-mode and is already
|
|
* connected.
|
|
* EADDRNOTAVAIL - The specified address is not available from the
|
|
* local machine.
|
|
* ECONNREFUSED - The target address was not listening for connections or
|
|
* refused the connection request because the connection backlog has
|
|
* been exceeded.
|
|
*
|
|
****************************************************************************/
|
|
|
|
int psock_local_connect(FAR struct socket *psock,
|
|
FAR const struct sockaddr *addr)
|
|
{
|
|
FAR struct local_conn_s *client;
|
|
FAR struct sockaddr_un *unaddr = (FAR struct sockaddr_un *)addr;
|
|
FAR struct local_conn_s *conn;
|
|
|
|
DEBUGASSERT(psock && psock->s_conn);
|
|
client = (FAR struct local_conn_s *)psock->s_conn;
|
|
|
|
if (client->lc_state == LOCAL_STATE_ACCEPT ||
|
|
client->lc_state == LOCAL_STATE_CONNECTED)
|
|
{
|
|
return -EISCONN;
|
|
}
|
|
|
|
/* Find the matching server connection */
|
|
|
|
net_lock();
|
|
for (conn = (FAR struct local_conn_s *)g_local_listeners.head;
|
|
conn;
|
|
conn = (FAR struct local_conn_s *)dq_next(&conn->lc_node))
|
|
{
|
|
/* Anything in the listener list should be a stream socket in the
|
|
* istening state
|
|
*/
|
|
|
|
DEBUGASSERT(conn->lc_state == LOCAL_STATE_LISTENING &&
|
|
conn->lc_proto == SOCK_STREAM);
|
|
|
|
/* Handle according to the server connection type */
|
|
|
|
switch (conn->lc_type)
|
|
{
|
|
case LOCAL_TYPE_UNNAMED: /* A Unix socket that is not bound to any name */
|
|
case LOCAL_TYPE_ABSTRACT: /* lc_path is length zero */
|
|
{
|
|
#warning Missing logic
|
|
net_unlock();
|
|
return OK;
|
|
}
|
|
break;
|
|
|
|
case LOCAL_TYPE_PATHNAME: /* lc_path holds a null terminated string */
|
|
{
|
|
if (strncmp(conn->lc_path, unaddr->sun_path, UNIX_PATH_MAX - 1)
|
|
== 0)
|
|
{
|
|
int ret = OK;
|
|
|
|
/* Bind the address and protocol */
|
|
|
|
client->lc_proto = conn->lc_proto;
|
|
strncpy(client->lc_path, unaddr->sun_path, UNIX_PATH_MAX - 1);
|
|
client->lc_path[UNIX_PATH_MAX - 1] = '\0';
|
|
client->lc_instance_id = local_generate_instance_id();
|
|
|
|
/* The client is now bound to an address */
|
|
|
|
client->lc_state = LOCAL_STATE_BOUND;
|
|
|
|
/* We have to do more for the SOCK_STREAM family */
|
|
|
|
if (conn->lc_proto == SOCK_STREAM)
|
|
{
|
|
ret = local_stream_connect(client, conn,
|
|
_SS_ISNONBLOCK(psock->s_flags));
|
|
}
|
|
else
|
|
{
|
|
net_unlock();
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
}
|
|
break;
|
|
|
|
default: /* Bad, memory must be corrupted */
|
|
DEBUGPANIC(); /* PANIC if debug on, else fall through */
|
|
|
|
case LOCAL_TYPE_UNTYPED: /* Type is not determined until the socket is bound */
|
|
{
|
|
net_unlock();
|
|
return -EINVAL;
|
|
}
|
|
}
|
|
}
|
|
|
|
net_unlock();
|
|
return -EADDRNOTAVAIL;
|
|
}
|
|
|
|
#endif /* CONFIG_NET_LOCAL_STREAM */
|