| /**************************************************************************** |
| * net/local/local_recvutils.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 <nuttx/config.h> |
| |
| #include <sys/types.h> |
| #include <stdint.h> |
| #include <unistd.h> |
| #include <string.h> |
| #include <errno.h> |
| #include <assert.h> |
| #include <debug.h> |
| |
| #include <nuttx/fs/fs.h> |
| |
| #include "local/local.h" |
| |
| /**************************************************************************** |
| * Public Functions |
| ****************************************************************************/ |
| |
| /**************************************************************************** |
| * Name: local_fifo_read |
| * |
| * Description: |
| * Read a data from the read-only FIFO. |
| * |
| * Input Parameters: |
| * filep - File structure of write-only FIFO. |
| * buf - Local to store the received data |
| * len - Length of data to receive [in] |
| * Length of data actually received [out] |
| * Zero means *len[in] is zero, |
| * or the sending side has closed the FIFO |
| * once - Flag to indicate the buf may only be read once |
| * |
| * Returned Value: |
| * Zero is returned on success; a negated errno value is returned on any |
| * failure. |
| * |
| ****************************************************************************/ |
| |
| int local_fifo_read(FAR struct file *filep, FAR uint8_t *buf, |
| size_t *len, bool once) |
| { |
| ssize_t remaining; |
| ssize_t nread; |
| int ret; |
| |
| DEBUGASSERT(buf && len); |
| |
| remaining = *len; |
| while (remaining > 0) |
| { |
| nread = file_read(filep, buf, remaining); |
| if (nread < 0) |
| { |
| ret = (int)nread; |
| |
| if (ret == -EINTR) |
| { |
| ninfo("Ignoring signal\n"); |
| continue; |
| } |
| else if (ret == -EAGAIN) |
| { |
| goto errout; |
| } |
| else |
| { |
| nerr("ERROR: file_read() failed: %d\n", ret); |
| goto errout; |
| } |
| } |
| else if (nread == 0) |
| { |
| /* The FIFO returns zero if the sending side of the connection |
| * has closed the FIFO. |
| */ |
| |
| break; |
| } |
| else |
| { |
| DEBUGASSERT(nread <= remaining); |
| remaining -= nread; |
| buf += nread; |
| |
| if (once) |
| { |
| break; |
| } |
| } |
| } |
| |
| ret = OK; |
| |
| errout: |
| *len -= remaining; |
| return ret; |
| } |
| |
| /**************************************************************************** |
| * Name: local_getaddr |
| * |
| * Description: |
| * Return the Unix domain address of a connection. |
| * |
| * Input Parameters: |
| * conn - The connection |
| * addr - The location to return the address |
| * addrlen - The size of the memory allocated by the caller to receive the |
| * address. |
| * |
| * Returned Value: |
| * Zero (OK) on success; a negated errno value on failure. |
| * |
| ****************************************************************************/ |
| |
| int local_getaddr(FAR struct local_conn_s *conn, FAR struct sockaddr *addr, |
| FAR socklen_t *addrlen) |
| { |
| FAR struct sockaddr_un *unaddr; |
| int totlen; |
| int pathlen; |
| |
| DEBUGASSERT(conn && addr && addrlen); |
| |
| /* Get the length of the path (minus the NUL terminator) and the length |
| * of the whole Unix domain address. |
| */ |
| |
| pathlen = strnlen(conn->lc_path, UNIX_PATH_MAX - 1); |
| totlen = sizeof(sa_family_t) + pathlen + 1; |
| |
| /* If the length of the whole Unix domain address is larger than the |
| * buffer provided by the caller, then truncate the address to fit. |
| */ |
| |
| if (totlen > *addrlen) |
| { |
| pathlen -= (totlen - *addrlen); |
| totlen = *addrlen; |
| } |
| |
| /* Copy the Unix domain address */ |
| |
| unaddr = (FAR struct sockaddr_un *)addr; |
| unaddr->sun_family = AF_LOCAL; |
| memcpy(unaddr->sun_path, conn->lc_path, pathlen); |
| unaddr->sun_path[pathlen] = '\0'; |
| |
| /* Return the Unix domain address size */ |
| |
| *addrlen = totlen; |
| return OK; |
| } |