2010-07-20 23:53:49 +04:00
|
|
|
/*
|
2018-03-21 23:48:51 +03:00
|
|
|
* Copyright (c) 2008-2018 Cisco Systems, Inc. All rights reserved
|
2010-07-20 23:53:49 +04:00
|
|
|
* Copyright (c) 2009 Sandia National Laboratories. All rights reserved.
|
2017-06-29 02:06:33 +03:00
|
|
|
* Copyright (c) 2017 Mellanox Technologies. All rights reserved.
|
2010-07-20 23:53:49 +04:00
|
|
|
*
|
|
|
|
* $COPYRIGHT$
|
2015-06-24 06:59:57 +03:00
|
|
|
*
|
2010-07-20 23:53:49 +04:00
|
|
|
* Additional copyrights may follow
|
2015-06-24 06:59:57 +03:00
|
|
|
*
|
2010-07-20 23:53:49 +04:00
|
|
|
* $HEADER$
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "opal_config.h"
|
|
|
|
|
2017-06-29 02:06:33 +03:00
|
|
|
#ifdef HAVE_SYS_TYPES_H
|
|
|
|
#include <sys/types.h>
|
|
|
|
#endif
|
|
|
|
#ifdef HAVE_SYS_STAT_H
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#endif
|
2018-03-21 23:48:51 +03:00
|
|
|
#ifdef HAVE_SYS_SOCKET_H
|
|
|
|
#include <sys/socket.h>
|
|
|
|
#endif
|
|
|
|
#ifdef HAVE_ARPA_INET_H
|
|
|
|
#include <arpa/inet.h>
|
|
|
|
#endif
|
2017-09-12 17:54:48 +03:00
|
|
|
#ifdef HAVE_NETINET_IN_H
|
|
|
|
#include <netinet/in.h>
|
|
|
|
#endif
|
2010-07-23 11:52:34 +04:00
|
|
|
#ifdef HAVE_UNISTD_H
|
2010-07-20 23:53:49 +04:00
|
|
|
#include <unistd.h>
|
2010-07-23 11:52:34 +04:00
|
|
|
#endif
|
2010-07-20 23:53:49 +04:00
|
|
|
#include <errno.h>
|
2014-04-24 17:20:32 +04:00
|
|
|
#include <fcntl.h>
|
2018-03-21 23:48:51 +03:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2010-07-20 23:53:49 +04:00
|
|
|
|
|
|
|
#include "opal/util/fd.h"
|
2018-09-27 20:31:12 +03:00
|
|
|
#include "opal/util/string_copy.h"
|
2010-07-20 23:53:49 +04:00
|
|
|
#include "opal/constants.h"
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Simple loop over reading from a fd
|
|
|
|
*/
|
|
|
|
int opal_fd_read(int fd, int len, void *buffer)
|
|
|
|
{
|
|
|
|
int rc;
|
|
|
|
char *b = buffer;
|
|
|
|
|
|
|
|
while (len > 0) {
|
|
|
|
rc = read(fd, b, len);
|
|
|
|
if (rc < 0 && (EAGAIN == errno || EINTR == errno)) {
|
|
|
|
continue;
|
|
|
|
} else if (rc > 0) {
|
|
|
|
len -= rc;
|
|
|
|
b += rc;
|
2010-08-24 23:07:04 +04:00
|
|
|
} else if (0 == rc) {
|
|
|
|
return OPAL_ERR_TIMEOUT;
|
2010-07-20 23:53:49 +04:00
|
|
|
} else {
|
|
|
|
return OPAL_ERR_IN_ERRNO;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return OPAL_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Simple loop over writing to an fd
|
|
|
|
*/
|
2010-07-21 15:01:16 +04:00
|
|
|
int opal_fd_write(int fd, int len, const void *buffer)
|
2010-07-20 23:53:49 +04:00
|
|
|
{
|
|
|
|
int rc;
|
2010-07-21 15:01:16 +04:00
|
|
|
const char *b = buffer;
|
2010-07-20 23:53:49 +04:00
|
|
|
|
|
|
|
while (len > 0) {
|
|
|
|
rc = write(fd, b, len);
|
|
|
|
if (rc < 0 && (EAGAIN == errno || EINTR == errno)) {
|
|
|
|
continue;
|
|
|
|
} else if (rc > 0) {
|
|
|
|
len -= rc;
|
|
|
|
b += rc;
|
|
|
|
} else {
|
|
|
|
return OPAL_ERR_IN_ERRNO;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return OPAL_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-24 17:04:49 +04:00
|
|
|
int opal_fd_set_cloexec(int fd)
|
|
|
|
{
|
|
|
|
#ifdef FD_CLOEXEC
|
|
|
|
int flags;
|
|
|
|
|
|
|
|
/* Stevens says that we should get the fd's flags before we set
|
|
|
|
them. So say we all. */
|
2014-04-24 17:20:32 +04:00
|
|
|
flags = fcntl(fd, F_GETFD, 0);
|
2014-04-24 17:04:49 +04:00
|
|
|
if (-1 == flags) {
|
|
|
|
return OPAL_ERR_IN_ERRNO;
|
|
|
|
}
|
|
|
|
|
2014-04-24 17:20:32 +04:00
|
|
|
if (fcntl(fd, F_SETFD, FD_CLOEXEC | flags) == -1) {
|
2014-04-24 17:04:49 +04:00
|
|
|
return OPAL_ERR_IN_ERRNO;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return OPAL_SUCCESS;
|
|
|
|
}
|
2017-06-29 02:06:33 +03:00
|
|
|
|
|
|
|
bool opal_fd_is_regular(int fd)
|
|
|
|
{
|
|
|
|
struct stat buf;
|
|
|
|
if (fstat(fd, &buf)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return S_ISREG(buf.st_mode);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool opal_fd_is_chardev(int fd)
|
|
|
|
{
|
|
|
|
struct stat buf;
|
|
|
|
if (fstat(fd, &buf)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return S_ISCHR(buf.st_mode);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool opal_fd_is_blkdev(int fd)
|
|
|
|
{
|
|
|
|
struct stat buf;
|
|
|
|
if (fstat(fd, &buf)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return S_ISBLK(buf.st_mode);
|
|
|
|
}
|
|
|
|
|
2018-03-21 23:48:51 +03:00
|
|
|
const char *opal_fd_get_peer_name(int fd)
|
|
|
|
{
|
|
|
|
char *str;
|
2018-03-24 14:25:26 +03:00
|
|
|
const char *ret = NULL;
|
2018-03-21 23:48:51 +03:00
|
|
|
struct sockaddr sa;
|
|
|
|
socklen_t slt = (socklen_t) sizeof(sa);
|
|
|
|
|
|
|
|
int rc = getpeername(fd, &sa, &slt);
|
|
|
|
if (0 != rc) {
|
|
|
|
ret = strdup("Unknown");
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t len = INET_ADDRSTRLEN;
|
|
|
|
#if OPAL_ENABLE_IPV6
|
|
|
|
len = INET6_ADDRSTRLEN;
|
|
|
|
#endif
|
2018-03-24 14:25:26 +03:00
|
|
|
str = calloc(1, len);
|
2018-03-21 23:48:51 +03:00
|
|
|
if (NULL == str) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sa.sa_family == AF_INET) {
|
|
|
|
struct sockaddr_in *si;
|
|
|
|
si = (struct sockaddr_in*) &sa;
|
|
|
|
ret = inet_ntop(AF_INET, &(si->sin_addr), str, INET_ADDRSTRLEN);
|
|
|
|
if (NULL == ret) {
|
|
|
|
free(str);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#if OPAL_ENABLE_IPV6
|
|
|
|
else if (sa.sa_family == AF_INET6) {
|
|
|
|
struct sockaddr_in6 *si6;
|
|
|
|
si6 = (struct sockaddr_in6*) &sa;
|
|
|
|
ret = inet_ntop(AF_INET6, &(si6->sin6_addr), str, INET6_ADDRSTRLEN);
|
|
|
|
if (NULL == ret) {
|
|
|
|
free(str);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
else {
|
2018-03-24 14:25:26 +03:00
|
|
|
// This string is guaranteed to be <= INET_ADDRSTRLEN
|
2018-09-27 20:31:12 +03:00
|
|
|
opal_string_copy(str, "Unknown", len);
|
2018-03-24 14:25:26 +03:00
|
|
|
ret = str;
|
2018-03-21 23:48:51 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|