summaryrefslogtreecommitdiff
path: root/sysdep/unix
diff options
context:
space:
mode:
authorMikael Magnusson <mikma@users.sourceforge.net>2019-03-27 23:18:25 +0100
committerMikael Magnusson <mikma@users.sourceforge.net>2023-08-28 21:06:52 +0200
commit67bf6bbdc1c1826ba9698e295e5ae96826663d9b (patch)
tree3ddcccf944ebfaf489d4cf2b81457681b02ff2c1 /sysdep/unix
parentf5140d1027f514bc59d46ab8aa09181f5870afbd (diff)
Unix: Implement sk_connect_unix
Support abstract socket in sk_unix_connect Implement SK_UNIX_ACTIVE
Diffstat (limited to 'sysdep/unix')
-rw-r--r--sysdep/unix/io.c59
-rw-r--r--sysdep/unix/unix.h1
2 files changed, 60 insertions, 0 deletions
diff --git a/sysdep/unix/io.c b/sysdep/unix/io.c
index 6aedcfb6..e5450fe9 100644
--- a/sysdep/unix/io.c
+++ b/sysdep/unix/io.c
@@ -1063,6 +1063,14 @@ sk_tcp_connected(sock *s)
s->tx_hook(s);
}
+static void
+sk_unix_connected(sock *s)
+{
+ sk_alloc_bufs(s);
+ s->type = SK_UNIX;
+ s->tx_hook(s);
+}
+
#ifdef HAVE_LIBSSH
static void
sk_ssh_connected(sock *s)
@@ -1529,6 +1537,51 @@ sk_open_unix(sock *s, char *name)
return 0;
}
+int
+sk_connect_unix(sock *s, char *name, socklen_t namelen)
+{
+ struct sockaddr_un sa;
+ int fd;
+
+ if (namelen > sizeof(sa.sun_path))
+ return -1;
+
+ fd = socket(AF_UNIX, SOCK_STREAM, 0);
+ if (fd < 0)
+ return -1;
+
+ if (fcntl(fd, F_SETFL, O_NONBLOCK) < 0)
+ {
+ close(fd);
+ return -3;
+ }
+
+ /* Path length checked in test_old_bird() */
+ memset(&sa, 0, sizeof(sa));
+ sa.sun_family = AF_UNIX;
+ memcpy(sa.sun_path, name, namelen);
+
+ s->fd = fd;
+ s->type = SK_UNIX_ACTIVE;
+ s->ttx = ""; /* Force s->ttx != s->tpos */
+
+ if (connect(fd, (struct sockaddr *) &sa, sizeof(sa.sun_family) + namelen) >= 0)
+ sk_unix_connected(s);
+ else if (errno != EINTR && errno != EAGAIN && errno != EINPROGRESS &&
+ errno != ECONNREFUSED && errno != EHOSTUNREACH && errno != ENETUNREACH)
+ {
+ ERR2("connect");
+ }
+
+ sk_insert(s);
+ return 0;
+
+ err:
+ close(fd);
+ s->fd = -1;
+ return -1;
+}
+
#define CMSG_RX_SPACE MAX(CMSG4_SPACE_PKTINFO+CMSG4_SPACE_TTL, \
CMSG6_SPACE_PKTINFO+CMSG6_SPACE_TTL)
@@ -1955,6 +2008,12 @@ sk_write_noflush(sock *s)
return 0;
}
+ case SK_UNIX_ACTIVE:
+ {
+ sk_unix_connected(s);
+ return 0;
+ }
+
#ifdef HAVE_LIBSSH
case SK_SSH_ACTIVE:
{
diff --git a/sysdep/unix/unix.h b/sysdep/unix/unix.h
index ad85d1ea..6fa8df6e 100644
--- a/sysdep/unix/unix.h
+++ b/sysdep/unix/unix.h
@@ -109,6 +109,7 @@ void io_loop(void);
void io_log_dump(void);
int sk_open_unix(struct birdsock *s, char *name);
struct rfile *rf_open(struct pool *, const char *name, const char *mode);
+int sk_connect_unix(struct birdsock *s, char *name, socklen_t namelen);
void *rf_file(struct rfile *f);
int rf_fileno(struct rfile *f);
void test_old_bird(char *path);