summaryrefslogtreecommitdiff
path: root/usr/src/lib/libcmd/common/fds.c
diff options
context:
space:
mode:
Diffstat (limited to 'usr/src/lib/libcmd/common/fds.c')
-rw-r--r--usr/src/lib/libcmd/common/fds.c168
1 files changed, 168 insertions, 0 deletions
diff --git a/usr/src/lib/libcmd/common/fds.c b/usr/src/lib/libcmd/common/fds.c
new file mode 100644
index 0000000000..b7e9fb2e88
--- /dev/null
+++ b/usr/src/lib/libcmd/common/fds.c
@@ -0,0 +1,168 @@
+/***********************************************************************
+* *
+* This software is part of the ast package *
+* Copyright (c) 1992-2007 AT&T Knowledge Ventures *
+* and is licensed under the *
+* Common Public License, Version 1.0 *
+* by AT&T Knowledge Ventures *
+* *
+* A copy of the License is available at *
+* http://www.opensource.org/licenses/cpl1.0.txt *
+* (with md5 checksum 059e8cd6165cb4c31e351f2b69388fd9) *
+* *
+* Information and Software Systems Research *
+* AT&T Research *
+* Florham Park NJ *
+* *
+* Glenn Fowler <gsf@research.att.com> *
+* David Korn <dgk@research.att.com> *
+* *
+***********************************************************************/
+#pragma prototyped
+
+static const char usage[] =
+"[-?\n@(#)$Id: fds (AT&T Research) 2006-10-26 $\n]"
+USAGE_LICENSE
+"[+NAME?fds - list open file descriptor status]"
+"[+DESCRIPTION?\bfds\b lists the status for each open file descriptor. "
+ "When invoked as a shell builtin it accesses the file descriptors of the "
+ "calling shell, otherwise it lists the file descriptors passed across "
+ "\bexec\b(2).]"
+"[l:long?List file descriptor details.]"
+"[+SEE ALSO?\blogname\b(1), \bwho\b(1), \bgetgroups\b(2)]"
+;
+
+#include <cmd.h>
+#include <ls.h>
+
+#include "FEATURE/sockets"
+
+#if defined(S_IFSOCK) && _sys_socket && _hdr_arpa_inet && _hdr_netinet_in && _lib_getsockname && _lib_getsockopt && _lib_inet_ntoa
+#include <sys/socket.h>
+#include <netinet/in.h>
+#include <arpa/inet.h>
+#else
+#undef S_IFSOCK
+#endif
+
+#ifndef minor
+#define minor(x) (int)((x)&0xff)
+#endif
+#ifndef major
+#define major(x) (int)(((unsigned int)(x)>>8)&0xff)
+#endif
+
+int
+b_fds(int argc, char** argv, void* context)
+{
+ register char* s;
+ register int i;
+ register char* m;
+ register char* x;
+ int flags;
+ int details;
+ struct stat st;
+#ifdef S_IFSOCK
+ struct sockaddr_in addr;
+ socklen_t len;
+ int type;
+ int prot;
+ char num[32];
+#endif
+
+ cmdinit(argc, argv, context, ERROR_CATALOG, 0);
+ details = 0;
+ for (;;)
+ {
+ switch (optget(argv, usage))
+ {
+ case 'l':
+ details = opt_info.num;
+ continue;
+ case '?':
+ error(ERROR_USAGE|4, "%s", opt_info.arg);
+ continue;
+ case ':':
+ error(2, "%s", opt_info.arg);
+ continue;
+ }
+ break;
+ }
+ argv += opt_info.index;
+ if (error_info.errors || *argv)
+ error(ERROR_USAGE|4, "%s", optusage(NiL));
+ for (i = 0; i <= OPEN_MAX; i++)
+ if (fstat(i, &st))
+ /* not open */;
+ else if (details)
+ {
+ if ((flags = fcntl(i, F_GETFL, (char*)0)) == -1)
+ m = "--";
+ else
+ switch (flags & (O_RDONLY|O_WRONLY|O_RDWR))
+ {
+ case O_RDONLY:
+ m = "r-";
+ break;
+ case O_WRONLY:
+ m = "-w";
+ break;
+ case O_RDWR:
+ m = "rw";
+ break;
+ default:
+ m = "??";
+ break;
+ }
+ x = (fcntl(i, F_GETFD, (char*)0) > 0) ? "x" : "-";
+ if (isatty(i) && (s = ttyname(i)))
+ sfprintf(sfstdout, "%02d %s%s %s %s\n", i, m, x, fmtmode(st.st_mode, 0), s);
+#ifdef S_IFSOCK
+ else if ((len = sizeof(addr))
+ && !getsockname(i, (struct sockaddr*)&addr, (void*)&len)
+ && len == sizeof(addr)
+ && addr.sin_family == AF_INET
+#ifdef SO_TYPE
+ && (len = sizeof(type))
+ && !getsockopt(i, SOL_SOCKET, SO_TYPE, (void*)&type, (void*)&len)
+ && len == sizeof(type)
+#else
+ && !(type = 0)
+#endif
+#ifdef SO_PROTOTYPE
+ && (len = sizeof(prot))
+ && (!getsockopt(i, SOL_SOCKET, SO_PROTOTYPE, (void*)&prot, (void*)&len) || !(prot = 0))
+#else
+ && !(prot = 0)
+#endif
+ )
+ {
+ if (!st.st_mode)
+ st.st_mode = S_IFSOCK|S_IRUSR|S_IWUSR;
+ s = 0;
+ switch (type)
+ {
+ case SOCK_DGRAM:
+ s = "udp";
+ break;
+ case SOCK_STREAM:
+ if (prot == 0)
+ s = "tcp";
+#ifdef IPPROTO_SCTP
+ else if (prot == IPPROTO_SCTP)
+ s = "sctp";
+#endif
+ break;
+ }
+ if (!s)
+ sfprintf(sfstdout, s = num, "type.%d.prot.%d", type, prot);
+ sfprintf(sfstdout, "%02d %s%s %s /dev/%s/%s/%d\n", i, m, x, fmtmode(st.st_mode, 0), s, inet_ntoa(addr.sin_addr), ntohs(addr.sin_port));
+ }
+#endif
+ else
+ sfprintf(sfstdout, "%02d %s%s %s /dev/inode/%u/%u\n", i, m, x, fmtmode(st.st_mode, 0), st.st_dev, st.st_ino);
+ }
+ else
+ sfprintf(sfstdout, "%d\n", i);
+ return 0;
+}