summaryrefslogtreecommitdiff
path: root/usr.bin/netstat/atalk.c
diff options
context:
space:
mode:
Diffstat (limited to 'usr.bin/netstat/atalk.c')
-rw-r--r--usr.bin/netstat/atalk.c333
1 files changed, 333 insertions, 0 deletions
diff --git a/usr.bin/netstat/atalk.c b/usr.bin/netstat/atalk.c
new file mode 100644
index 00000000000..f6b33c45fb1
--- /dev/null
+++ b/usr.bin/netstat/atalk.c
@@ -0,0 +1,333 @@
+/* $NetBSD: atalk.c,v 1.2 1997/05/22 17:21:26 christos Exp $ */
+
+/*
+ * Copyright (c) 1983, 1988, 1993
+ * The Regents of the University of California. All rights reserved.
+ *
+ * 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. All advertising materials mentioning features or use of this software
+ * must display the following acknowledgement:
+ * This product includes software developed by the University of
+ * California, Berkeley and its contributors.
+ * 4. Neither the name of the University 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 REGENTS 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 REGENTS 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.
+ */
+
+#ifndef lint
+#if 0
+static char sccsid[] = "from @(#)atalk.c 1.1 (Whistle) 6/6/96";
+#else
+static char rcsid[] = "$NetBSD: atalk.c,v 1.2 1997/05/22 17:21:26 christos Exp $";
+#endif
+#endif /* not lint */
+
+#include <sys/param.h>
+#include <sys/queue.h>
+#include <sys/socket.h>
+#include <sys/socketvar.h>
+#include <sys/mbuf.h>
+#include <sys/protosw.h>
+#include <netdb.h>
+
+#include <net/route.h>
+#include <net/if.h>
+
+/* #include <netinet/tcp_fsm.h> */
+
+#include <netatalk/at.h>
+#include <netatalk/ddp_var.h>
+
+#include <nlist.h>
+#include <errno.h>
+#include <stdio.h>
+#include <string.h>
+#include "netstat.h"
+
+struct ddpcb ddpcb;
+struct socket sockb;
+
+static int first = 1;
+
+static char *at_pr_net __P((struct sockaddr_at *, int));
+static char *at_pr_host __P((struct sockaddr_at *, int));
+static char *at_pr_range __P((struct sockaddr_at *));
+static char *at_pr_port __P((struct sockaddr_at *));
+
+/*
+ * Print a summary of connections related to a Network Systems
+ * protocol. For XXX, also give state of connection.
+ * Listening processes (aflag) are suppressed unless the
+ * -a (all) flag is specified.
+ */
+
+static char *
+at_pr_net(sat, numeric)
+ struct sockaddr_at *sat;
+ int numeric;
+{
+ static char mybuf[50];
+
+ if (!numeric) {
+ switch (sat->sat_addr.s_net) {
+ case 0xffff:
+ return "????";
+ case ATADDR_ANYNET:
+ return ("*");
+ }
+ }
+ (void) snprintf(mybuf, sizeof(mybuf), "%hu",
+ ntohs(sat->sat_addr.s_net));
+ return mybuf;
+}
+
+static char *
+at_pr_host(sat, numeric)
+ struct sockaddr_at *sat;
+ int numeric;
+{
+ static char mybuf[50];
+
+ if (!numeric) {
+ switch (sat->sat_addr.s_node) {
+ case ATADDR_BCAST:
+ return "bcast";
+ case ATADDR_ANYNODE:
+ return ("*");
+ }
+ }
+ (void) snprintf(mybuf, sizeof(mybuf), "%d",
+ (unsigned int) sat->sat_addr.s_node);
+ return mybuf;
+}
+
+static char *
+at_pr_port(sat)
+ struct sockaddr_at *sat;
+{
+ static char mybuf[50];
+ struct servent *serv;
+
+ switch (sat->sat_port) {
+ case ATADDR_ANYPORT:
+ return ("*");
+ case 0xff:
+ return "????";
+ default:
+ if (nflag)
+ (void) snprintf(mybuf, sizeof(mybuf), "%d",
+ (unsigned int) sat->sat_port);
+ else {
+ serv = getservbyport(sat->sat_port, "ddp");
+ if (serv == NULL)
+ (void) snprintf(mybuf, sizeof(mybuf), "%d",
+ (unsigned int) sat->sat_port);
+ else
+ (void) snprintf(mybuf, sizeof(mybuf), "%s",
+ serv->s_name);
+ }
+
+ return mybuf;
+ }
+}
+
+static char *
+at_pr_range(sat)
+ struct sockaddr_at *sat;
+{
+ static char mybuf[50];
+
+ if (sat->sat_range.r_netrange.nr_firstnet
+ != sat->sat_range.r_netrange.nr_lastnet) {
+ (void) snprintf(mybuf, sizeof(mybuf), "%d-%d",
+ ntohs(sat->sat_range.r_netrange.nr_firstnet),
+ ntohs(sat->sat_range.r_netrange.nr_lastnet));
+ } else {
+ (void) snprintf(mybuf, sizeof(mybuf), "%d",
+ ntohs(sat->sat_range.r_netrange.nr_firstnet));
+ }
+ return mybuf;
+}
+
+
+/* what == 0 for addr only == 3
+ * 1 for net
+ * 2 for host
+ * 4 for port
+ * 8 for numeric only
+ */
+char *
+atalk_print(sa, what)
+ const struct sockaddr *sa;
+ int what;
+{
+ struct sockaddr_at *sat = (struct sockaddr_at *) sa;
+ static char mybuf[50];
+ int numeric = (what & 0x08);
+
+ mybuf[0] = 0;
+ switch (what & 0x13) {
+ case 0:
+ mybuf[0] = 0;
+ break;
+ case 1:
+ (void) snprintf(mybuf, sizeof(mybuf), "%s",
+ at_pr_net(sat, numeric));
+ break;
+ case 2:
+ (void) snprintf(mybuf, sizeof(mybuf), "%s",
+ at_pr_host(sat, numeric));
+ break;
+ case 3:
+ (void) snprintf(mybuf, sizeof(mybuf), "%s.%s",
+ at_pr_net(sat, numeric),
+ at_pr_host(sat, numeric));
+ break;
+ case 0x10:
+ (void) snprintf(mybuf, sizeof(mybuf), "%s", at_pr_range(sat));
+ }
+ if (what & 4) {
+ (void) snprintf(mybuf + strlen(mybuf),
+ sizeof(mybuf) - strlen(mybuf), ".%s",
+ at_pr_port(sat));
+ }
+ return mybuf;
+}
+
+char *
+atalk_print2(sa, mask, what)
+ const struct sockaddr *sa;
+ const struct sockaddr *mask;
+ int what;
+{
+ int n;
+ static char buf[100];
+ struct sockaddr_at *sat1, *sat2;
+ struct sockaddr_at thesockaddr;
+ struct sockaddr *sa2;
+
+ sat1 = (struct sockaddr_at *) sa;
+ sat2 = (struct sockaddr_at *) mask;
+ sa2 = (struct sockaddr *) & thesockaddr;
+
+ thesockaddr.sat_addr.s_net = sat1->sat_addr.s_net &
+ sat2->sat_addr.s_net;
+ n = snprintf(buf, sizeof(buf), "%s", atalk_print(sa2, 1 | (what & 8)));
+ if (sat2->sat_addr.s_net != 0xFFFF) {
+ thesockaddr.sat_addr.s_net = sat1->sat_addr.s_net |
+ ~sat2->sat_addr.s_net;
+ n += snprintf(buf + n, sizeof(buf) - n,
+ "-%s", atalk_print(sa2, 1 | (what & 8)));
+ }
+ if (what & 2)
+ n += snprintf(buf + n, sizeof(buf) - n, ".%s",
+ atalk_print(sa, what & (~1)));
+ return (buf);
+}
+
+void
+atalkprotopr(off, name)
+ u_long off;
+ char *name;
+{
+ struct ddpcb cb;
+ register struct ddpcb *prev, *next;
+ struct ddpcb *initial;
+
+ if (off == 0)
+ return;
+ if (kread(off, (char *) &initial, sizeof(struct ddpcb *)) < 0)
+ return;
+ ddpcb = cb;
+ prev = (struct ddpcb *) off;
+ for (next = initial; next != NULL; prev = next) {
+ u_long ppcb = (u_long) next;
+
+ if (kread((u_long) next, (char *) &ddpcb, sizeof(ddpcb)) < 0)
+ return;
+ next = ddpcb.ddp_next;
+#if 0
+ if (!aflag && atalk_nullhost(ddpcb.ddp_lsat)) {
+ continue;
+ }
+#endif
+ if (kread((u_long) ddpcb.ddp_socket,
+ (char *) &sockb, sizeof(sockb)) < 0)
+ return;
+ if (first) {
+ printf("Active ATALK connections");
+ if (aflag)
+ printf(" (including servers)");
+ putchar('\n');
+ if (Aflag)
+ printf("%-8.8s ", "PCB");
+ printf(Aflag ?
+ "%-5.5s %-6.6s %-6.6s %-18.18s %-18.18s %s\n" :
+ "%-5.5s %-6.6s %-6.6s %-22.22s %-22.22s %s\n",
+ "Proto", "Recv-Q", "Send-Q",
+ "Local Address", "Foreign Address", "(state)");
+ first = 0;
+ }
+ if (Aflag)
+ printf("%8lx ", ppcb);
+ printf("%-5.5s %6ld %6ld ", name, sockb.so_rcv.sb_cc,
+ sockb.so_snd.sb_cc);
+ printf(Aflag ? " %-18.18s" : " %-22.22s", atalk_print(
+ (struct sockaddr *) & ddpcb.ddp_lsat, 7));
+ printf(Aflag ? " %-18.18s" : " %-22.22s", atalk_print(
+ (struct sockaddr *) & ddpcb.ddp_fsat, 7));
+ putchar('\n');
+ }
+}
+
+#define p(f, m) if (ddpstat.f || sflag <= 1) \
+ printf(m, ddpstat.f, plural(ddpstat.f))
+#define p2(f1, f2, m) if (ddpstat.f1 || ddpstat.f2 || sflag <= 1) \
+ printf(m, ddpstat.f1, plural(ddpstat.f1), ddpstat.f2, plural(ddpstat.f2))
+#define p3(f, m) if (ddpstat.f || sflag <= 1) \
+ printf(m, ddpstat.f, plurales(ddpstat.f))
+
+/*
+ * Dump DDP statistics structure.
+ */
+void
+ddp_stats(off, name)
+ u_long off;
+ char *name;
+{
+ struct ddpstat ddpstat;
+
+ if (off == 0)
+ return;
+ if (kread(off, (char *) &ddpstat, sizeof(ddpstat)) < 0)
+ return;
+ printf("%s:\n", name);
+ p(ddps_short, "\t%ld packet%s with short headers\n");
+ p(ddps_long, "\t%ld packet%s with long headers\n");
+ p(ddps_nosum, "\t%ld packet%s with no checksum\n");
+ p(ddps_tooshort, "\t%ld packet%s were too short\n");
+ p(ddps_badsum, "\t%ld packet%s with bad checksum\n");
+ p(ddps_toosmall, "\t%ld packet%s with not enough data\n");
+ p(ddps_forward, "\t%ld packet%s forwarded\n");
+ p(ddps_cantforward, "\t%ld packet%s rcvd for unreachable dest\n");
+ p(ddps_nosockspace, "\t%ld packet%s dropped due to no socket space\n");
+}