sockstat(1): use tree(3) rbtree instead of hash
o Use tree to lookup by socket kvaddr. The size of hash is too big for a small virtual machine and at the same time too little for a large production server. A tree would better fit here. o For those pcbs, that don't have a socket associated, use a list. o Provide a second tree to lookup by pcb kvaddr. These removes full hash traversal when printing every unix(4) socket. Reviewed by: tuexen Differential revision: https://reviews.freebsd.org/D35724
This commit is contained in:
parent
7d016011f4
commit
a83d596f43
@ -38,6 +38,8 @@ __FBSDID("$FreeBSD$");
|
||||
#include <sys/sysctl.h>
|
||||
#include <sys/jail.h>
|
||||
#include <sys/user.h>
|
||||
#include <sys/queue.h>
|
||||
#include <sys/tree.h>
|
||||
|
||||
#include <sys/un.h>
|
||||
#include <sys/unpcb.h>
|
||||
@ -119,6 +121,11 @@ struct addr {
|
||||
};
|
||||
|
||||
struct sock {
|
||||
union {
|
||||
RB_ENTRY(sock) socket_tree; /* tree of pcbs with socket */
|
||||
SLIST_ENTRY(sock) socket_list; /* list of pcbs w/o socket */
|
||||
};
|
||||
RB_ENTRY(sock) pcb_tree;
|
||||
kvaddr_t socket;
|
||||
kvaddr_t pcb;
|
||||
uint64_t inp_gencnt;
|
||||
@ -132,11 +139,25 @@ struct sock {
|
||||
char cc[TCP_CA_NAME_MAX];
|
||||
struct addr *laddr;
|
||||
struct addr *faddr;
|
||||
struct sock *next;
|
||||
};
|
||||
|
||||
#define HASHSIZE 1009
|
||||
static struct sock *sockhash[HASHSIZE];
|
||||
static RB_HEAD(socks_t, sock) socks = RB_INITIALIZER(&socks);
|
||||
static int64_t
|
||||
socket_compare(const struct sock *a, const struct sock *b)
|
||||
{
|
||||
return ((int64_t)(a->socket/2 - b->socket/2));
|
||||
}
|
||||
RB_GENERATE_STATIC(socks_t, sock, socket_tree, socket_compare);
|
||||
|
||||
static RB_HEAD(pcbs_t, sock) pcbs = RB_INITIALIZER(&pcbs);
|
||||
static int64_t
|
||||
pcb_compare(const struct sock *a, const struct sock *b)
|
||||
{
|
||||
return ((int64_t)(a->pcb/2 - b->pcb/2));
|
||||
}
|
||||
RB_GENERATE_STATIC(pcbs_t, sock, pcb_tree, pcb_compare);
|
||||
|
||||
static SLIST_HEAD(, sock) nosocks = SLIST_HEAD_INITIALIZER(&nosocks);
|
||||
|
||||
static struct xfile *xfiles;
|
||||
static int nxfiles;
|
||||
@ -350,7 +371,7 @@ gather_sctp(void)
|
||||
const char *varname;
|
||||
size_t len, offset;
|
||||
char *buf;
|
||||
int hash, vflag;
|
||||
int vflag;
|
||||
int no_stcb, local_all_loopback, foreign_all_loopback;
|
||||
|
||||
vflag = 0;
|
||||
@ -469,10 +490,7 @@ gather_sctp(void)
|
||||
(!opt_L || !local_all_loopback) &&
|
||||
((xinpcb->flags & SCTP_PCB_FLAGS_UDPTYPE) ||
|
||||
(xstcb->last == 1))) {
|
||||
hash = (int)((uintptr_t)sock->socket %
|
||||
HASHSIZE);
|
||||
sock->next = sockhash[hash];
|
||||
sockhash[hash] = sock;
|
||||
RB_INSERT(socks_t, &socks, sock);
|
||||
} else {
|
||||
free_socket(sock);
|
||||
}
|
||||
@ -597,10 +615,7 @@ gather_sctp(void)
|
||||
(!opt_L ||
|
||||
!(local_all_loopback ||
|
||||
foreign_all_loopback))) {
|
||||
hash = (int)((uintptr_t)sock->socket %
|
||||
HASHSIZE);
|
||||
sock->next = sockhash[hash];
|
||||
sockhash[hash] = sock;
|
||||
RB_INSERT(socks_t, &socks, sock);
|
||||
} else {
|
||||
free_socket(sock);
|
||||
}
|
||||
@ -624,7 +639,7 @@ gather_inet(int proto)
|
||||
const char *varname, *protoname;
|
||||
size_t len, bufsize;
|
||||
void *buf;
|
||||
int hash, retry, vflag;
|
||||
int retry, vflag;
|
||||
|
||||
vflag = 0;
|
||||
if (opt_4)
|
||||
@ -760,9 +775,10 @@ gather_inet(int proto)
|
||||
memcpy(sock->cc, xtp->xt_cc, TCP_CA_NAME_MAX);
|
||||
}
|
||||
sock->protoname = protoname;
|
||||
hash = (int)((uintptr_t)sock->socket % HASHSIZE);
|
||||
sock->next = sockhash[hash];
|
||||
sockhash[hash] = sock;
|
||||
if (sock->socket != 0)
|
||||
RB_INSERT(socks_t, &socks, sock);
|
||||
else
|
||||
SLIST_INSERT_HEAD(&nosocks, sock, socket_list);
|
||||
}
|
||||
out:
|
||||
free(buf);
|
||||
@ -778,7 +794,7 @@ gather_unix(int proto)
|
||||
const char *varname, *protoname;
|
||||
size_t len, bufsize;
|
||||
void *buf;
|
||||
int hash, retry;
|
||||
int retry;
|
||||
|
||||
switch (proto) {
|
||||
case SOCK_STREAM:
|
||||
@ -852,9 +868,8 @@ gather_unix(int proto)
|
||||
faddr->next = NULL;
|
||||
sock->laddr = laddr;
|
||||
sock->faddr = faddr;
|
||||
hash = (int)((uintptr_t)sock->socket % HASHSIZE);
|
||||
sock->next = sockhash[hash];
|
||||
sockhash[hash] = sock;
|
||||
RB_INSERT(socks_t, &socks, sock);
|
||||
RB_INSERT(pcbs_t, &pcbs, sock);
|
||||
}
|
||||
out:
|
||||
free(buf);
|
||||
@ -1052,7 +1067,7 @@ static void
|
||||
displaysock(struct sock *s, int pos)
|
||||
{
|
||||
kvaddr_t p;
|
||||
int hash, first, offset;
|
||||
int first, offset;
|
||||
struct addr *laddr, *faddr;
|
||||
struct sock *s_tmp;
|
||||
|
||||
@ -1104,15 +1119,8 @@ displaysock(struct sock *s, int pos)
|
||||
break;
|
||||
}
|
||||
pos += xprintf("-> ");
|
||||
for (hash = 0; hash < HASHSIZE; ++hash) {
|
||||
for (s_tmp = sockhash[hash];
|
||||
s_tmp != NULL;
|
||||
s_tmp = s_tmp->next)
|
||||
if (s_tmp->pcb == p)
|
||||
break;
|
||||
if (s_tmp != NULL)
|
||||
break;
|
||||
}
|
||||
s_tmp = RB_FIND(pcbs_t, &pcbs,
|
||||
&(struct sock){ .pcb = p });
|
||||
if (s_tmp == NULL || s_tmp->laddr == NULL ||
|
||||
s_tmp->laddr->address.ss_len == 0)
|
||||
pos += xprintf("??");
|
||||
@ -1222,7 +1230,7 @@ display(void)
|
||||
struct passwd *pwd;
|
||||
struct xfile *xf;
|
||||
struct sock *s;
|
||||
int hash, n, pos;
|
||||
int n, pos;
|
||||
|
||||
if (opt_q != 1) {
|
||||
printf("%-8s %-10s %-5s %-2s %-6s %-*s %-*s",
|
||||
@ -1250,12 +1258,9 @@ display(void)
|
||||
continue;
|
||||
if (opt_j >= 0 && opt_j != getprocjid(xf->xf_pid))
|
||||
continue;
|
||||
hash = (int)((uintptr_t)xf->xf_data % HASHSIZE);
|
||||
for (s = sockhash[hash]; s != NULL; s = s->next) {
|
||||
if (s->socket != xf->xf_data)
|
||||
continue;
|
||||
if (!check_ports(s))
|
||||
continue;
|
||||
s = RB_FIND(socks_t, &socks,
|
||||
&(struct sock){ .socket = xf->xf_data});
|
||||
if (s != NULL && check_ports(s)) {
|
||||
s->shown = 1;
|
||||
pos = 0;
|
||||
if (opt_n ||
|
||||
@ -1277,19 +1282,23 @@ display(void)
|
||||
}
|
||||
if (opt_j >= 0)
|
||||
return;
|
||||
for (hash = 0; hash < HASHSIZE; hash++) {
|
||||
for (s = sockhash[hash]; s != NULL; s = s->next) {
|
||||
SLIST_FOREACH(s, &nosocks, socket_list) {
|
||||
if (!check_ports(s))
|
||||
continue;
|
||||
pos = xprintf("%-8s %-10s %-5s %-2s ",
|
||||
"?", "?", "?", "?");
|
||||
displaysock(s, pos);
|
||||
}
|
||||
RB_FOREACH(s, socks_t, &socks) {
|
||||
if (s->shown)
|
||||
continue;
|
||||
if (!check_ports(s))
|
||||
continue;
|
||||
pos = 0;
|
||||
pos += xprintf("%-8s %-10s %-5s %-2s ",
|
||||
pos = xprintf("%-8s %-10s %-5s %-2s ",
|
||||
"?", "?", "?", "?");
|
||||
displaysock(s, pos);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static int
|
||||
set_default_protos(void)
|
||||
|
Loading…
Reference in New Issue
Block a user