summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorEric Jackson <ericj@cvs.openbsd.org>2002-02-17 03:03:07 +0000
committerEric Jackson <ericj@cvs.openbsd.org>2002-02-17 03:03:07 +0000
commit31dad924e534efcc96721b9bc79b973d72135366 (patch)
tree5c4a9c05b6b06bd200f073b9538229a0295a37da
parent7d77b876dd5f005f98482cdb1d04ff6e5eebf65e (diff)
add support for connecting too and listening on AF_UNIX sockets.
connect support from dave@arbor.net.. rest by me
-rw-r--r--usr.bin/nc/nc.110
-rw-r--r--usr.bin/nc/netcat.c114
2 files changed, 111 insertions, 13 deletions
diff --git a/usr.bin/nc/nc.1 b/usr.bin/nc/nc.1
index 3025e99c4b9..721a2c48963 100644
--- a/usr.bin/nc/nc.1
+++ b/usr.bin/nc/nc.1
@@ -1,4 +1,4 @@
-.\" $OpenBSD: nc.1,v 1.19 2001/09/02 18:45:41 jakob Exp $
+.\" $OpenBSD: nc.1,v 1.20 2002/02/17 03:03:06 ericj Exp $
.\"
.\" Copyright (c) 1996 David Sacerdote
.\" All rights reserved.
@@ -33,7 +33,7 @@
.Nd "arbitrary TCP and UDP connections and listens"
.Sh SYNOPSIS
.Nm nc
-.Op Fl 46hklnrtuvz
+.Op Fl 46hklnrtuvzU
.Op Fl i Ar interval
.Op Fl p Ar source port
.Op Fl s Ar source ip address
@@ -139,6 +139,8 @@ If port is not specified, port 1080 is used.
Specifies that
.Nm
should just scan for listening daemons, without sending any data to them.
+.It Fl U
+Specifies to use Unix Domain Sockets.
.El
.Sh EXAMPLES
.Bl -tag -width x
@@ -176,6 +178,10 @@ the remote host, and send data from the remote host to stdout.
Connect to port 1000 of hostname, send the string "foobar"
followed by a newline, and move data from port 1000 of hostname to
stdout until hostname closes the connection.
+.It Li "nc -U /var/tmp/dsocket"
+Connect to a Unix Domain Socket.
+.It Li "nc -lU /var/tmp/dsocket"
+Create and listen on a Unix Domain Socket.
.El
.Sh SEE ALSO
.Xr cat 1 ,
diff --git a/usr.bin/nc/netcat.c b/usr.bin/nc/netcat.c
index 51d2501ccbb..f95eb16e099 100644
--- a/usr.bin/nc/netcat.c
+++ b/usr.bin/nc/netcat.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: netcat.c,v 1.41 2002/02/17 02:04:57 ericj Exp $ */
+/* $OpenBSD: netcat.c,v 1.42 2002/02/17 03:03:06 ericj Exp $ */
/*
* Copyright (c) 2001 Eric Jackson <ericj@monkey.org>
*
@@ -34,6 +34,7 @@
#include <sys/types.h>
#include <sys/socket.h>
#include <sys/time.h>
+#include <sys/un.h>
#include <netinet/in.h>
#include <arpa/telnet.h>
@@ -47,6 +48,7 @@
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
+#include <fcntl.h>
#define PORT_MAX 65535
@@ -76,6 +78,8 @@ int local_listen(char *, char *, struct addrinfo);
void readwrite(int);
int remote_connect(char *, char *, struct addrinfo);
int udptest(int);
+int unix_connect(char *);
+int unix_listen(char *);
void usage(int);
int
@@ -98,7 +102,7 @@ main(int argc, char *argv[])
endp = NULL;
sv = NULL;
- while ((ch = getopt(argc, argv, "46hi:klnp:rs:tuvw:x:z")) != -1) {
+ while ((ch = getopt(argc, argv, "46Uhi:klnp:rs:tuvw:x:z")) != -1) {
switch (ch) {
case '4':
family = AF_INET;
@@ -106,6 +110,9 @@ main(int argc, char *argv[])
case '6':
family = AF_INET6;
break;
+ case 'U':
+ family = AF_UNIX;
+ break;
case 'h':
help();
break;
@@ -161,7 +168,12 @@ main(int argc, char *argv[])
argv += optind;
/* Cruft to make sure options are clean, and used properly. */
- if (argv[0] && !argv[1]) {
+ if (argv[0] && !argv[1] && family == AF_UNIX) {
+ if (uflag)
+ errx(1, "cannot use -u and -U");
+ host = argv[0];
+ uport = NULL;
+ } else if (argv[0] && !argv[1]) {
if (!lflag)
usage(1);
uport = argv[0];
@@ -182,12 +194,15 @@ main(int argc, char *argv[])
errx(1, "must use -l with -k");
/* Initialize addrinfo structure */
- memset(&hints, 0, sizeof(struct addrinfo));
- hints.ai_family = family;
- hints.ai_socktype = uflag ? SOCK_DGRAM : SOCK_STREAM;
- hints.ai_protocol = uflag ? IPPROTO_UDP : IPPROTO_TCP;
- if (nflag)
- hints.ai_flags |= AI_NUMERICHOST;
+ if (family != AF_UNIX) {
+ memset(&hints, 0, sizeof(struct addrinfo));
+ hints.ai_family = family;
+ hints.ai_socktype = uflag ? SOCK_DGRAM : SOCK_STREAM;
+ hints.ai_protocol = uflag ? IPPROTO_UDP : IPPROTO_TCP;
+ if (nflag)
+ hints.ai_flags |= AI_NUMERICHOST;
+ }
+
if (xflag) {
if (uflag)
@@ -196,6 +211,9 @@ main(int argc, char *argv[])
if (lflag)
errx(1, "no proxy support for listen");
+ if (family == AF_UNIX)
+ errx(1, "no proxy support for unix sockets");
+
/* XXX IPv6 transport to proxy would probably work */
if (family == AF_INET6)
errx(1, "no proxy support for IPv6");
@@ -218,9 +236,14 @@ main(int argc, char *argv[])
int connfd;
ret = 0;
+ if (family == AF_UNIX)
+ s = unix_listen(host);
+
/* Allow only one connection at a time, but stay alive */
for (;;) {
- if ((s = local_listen(host, uport, hints)) < 0)
+ if (family != AF_UNIX)
+ s = local_listen(host, uport, hints);
+ if (s < 0)
err(1, NULL);
/*
* For UDP, we will use recvfrom() initially
@@ -250,11 +273,23 @@ main(int argc, char *argv[])
readwrite(connfd);
close(connfd);
- close(s);
+ if (family != AF_UNIX)
+ close(s);
if (!kflag)
break;
}
+ } else if (family == AF_UNIX) {
+ ret = 0;
+
+ if ((s = unix_connect(host)) > 0 && !zflag) {
+ readwrite(s);
+ close(s);
+ } else
+ ret = 1;
+
+ exit(ret);
+
} else {
int i = 0;
@@ -311,6 +346,62 @@ main(int argc, char *argv[])
}
/*
+ * unix_connect()
+ * Return's a socket connected to a local unix socket. Return's -1 on failure.
+ */
+int
+unix_connect(char *path)
+{
+ struct sockaddr_un sun;
+ int s;
+
+ if ((s = socket(AF_UNIX, SOCK_STREAM, 0)) < 0)
+ return (-1);
+ (void)fcntl(s, F_SETFD, 1);
+
+ memset(&sun, 0, sizeof(struct sockaddr_un));
+ sun.sun_len = sizeof(path);
+ sun.sun_family = AF_UNIX;
+ strlcpy(sun.sun_path, path, sizeof(sun.sun_path));
+
+ if (connect(s, (struct sockaddr *)&sun, sizeof(sun)) < 0) {
+ close(s);
+ return (-1);
+ }
+ return (s);
+
+}
+
+/*
+ * unix_listen()
+ * create a unix domain socket, and listen on it.
+ */
+int
+unix_listen(char *path)
+{
+ struct sockaddr_un sun;
+ int s;
+
+ /* create unix domain socket */
+ if ((s = socket(AF_UNIX, SOCK_STREAM, 0)) < 0)
+ return (-1);
+
+ sun.sun_family = AF_UNIX;
+ strlcpy(sun.sun_path, path, sizeof(sun.sun_path));
+
+ if (bind(s, (struct sockaddr *)&sun, SUN_LEN(&sun)) < 0) {
+ close(s);
+ return (-1);
+ }
+
+ if (listen(s, 5) < 0) {
+ close(s);
+ return (-1);
+ }
+ return (s);
+}
+
+/*
* remote_connect()
* Return's a socket connected to a remote host. Properly bind's to a local
* port or source address if needed. Return's -1 on failure.
@@ -598,6 +689,7 @@ help()
fprintf(stderr, "\tCommand Summary:\n\
\t-4 Use IPv4\n\
\t-6 Use IPv6\n\
+ \t-U Use UNIX domain socket\n\
\t-h This help text\n\
\t-i secs\t Delay interval for lines sent, ports scanned\n\
\t-k Keep inbound sockets open for multiple connects\n\