Skip to content

Commit

Permalink
[FreeBSD] speed up net_connections() (#2343)
Browse files Browse the repository at this point in the history
  • Loading branch information
giampaolo committed Dec 20, 2023
1 parent 2786c09 commit 13e1fe7
Show file tree
Hide file tree
Showing 3 changed files with 77 additions and 24 deletions.
13 changes: 4 additions & 9 deletions HISTORY.rst
Original file line number Diff line number Diff line change
Expand Up @@ -5,17 +5,12 @@

**Enhancements**

- 2342_, [NetBSD]: filter `net_connections()`_ returned list in C instead of
- 2343_, [FreeBSD]: filter `net_connections()`_ returned list in C instead of
Python, and avoid to retrieve unnecessary connection types unless explicitly
asked. E.g., on an IDLE system with few IPv6 connections this will run around
170% faster. Before all connection types (TCP, UDP, UNIX) were retrived
internally, even if they were not returned.::

import psutil, time
started = time.monotonic()
for x in range(1000):
psutil.net_connections("tcp6")
print(f"completed in {(time.monotonic() - started):.4f} secs")
4 times faster. Before all connection types (TCP, UDP, UNIX) were retrieved
internally, even if only a portion was returned.
- 2342_, [NetBSD]: same as above (#2343) but for NetBSD.

**Bug fixes**

Expand Down
5 changes: 1 addition & 4 deletions psutil/_psbsd.py
Original file line number Diff line number Diff line change
Expand Up @@ -428,13 +428,10 @@ def net_connections(kind):
elif NETBSD:
rawlist = cext.net_connections(-1, kind)
else: # FreeBSD
rawlist = cext.net_connections()
rawlist = cext.net_connections(families, types)

for item in rawlist:
fd, fam, type, laddr, raddr, status, pid = item
if FREEBSD:
if (fam not in families) or (type not in types):
continue
nt = conn_to_ntuple(fd, fam, type, laddr, raddr,
status, TCP_STATUSES, pid)
ret.add(nt)
Expand Down
83 changes: 72 additions & 11 deletions psutil/arch/freebsd/sys_socks.c
Original file line number Diff line number Diff line change
Expand Up @@ -33,7 +33,7 @@ static struct xfile *psutil_xfiles;
static int psutil_nxfiles;


int
static int
psutil_populate_xfiles(void) {
size_t len;

Expand Down Expand Up @@ -61,7 +61,7 @@ psutil_populate_xfiles(void) {
}


struct xfile *
static struct xfile *
psutil_get_file_from_sock(kvaddr_t sock) {
struct xfile *xf;
int n;
Expand All @@ -76,7 +76,10 @@ psutil_get_file_from_sock(kvaddr_t sock) {

// Reference:
// https://github.com/freebsd/freebsd/blob/master/usr.bin/sockstat/sockstat.c
int psutil_gather_inet(int proto, PyObject *py_retlist) {
static int
psutil_gather_inet(
int proto, int include_v4, int include_v6, PyObject *py_retlist)
{
struct xinpgen *xig, *exig;
struct xinpcb *xip;
struct xtcpcb *xtp;
Expand Down Expand Up @@ -177,6 +180,12 @@ int psutil_gather_inet(int proto, PyObject *py_retlist) {
goto error;
}

// filter
if ((inp->inp_vflag & INP_IPV4) && (include_v4 == 0))
continue;
if ((inp->inp_vflag & INP_IPV6) && (include_v6 == 0))
continue;

char lip[200], rip[200];

xf = psutil_get_file_from_sock(so->xso_so);
Expand Down Expand Up @@ -235,7 +244,8 @@ int psutil_gather_inet(int proto, PyObject *py_retlist) {
}


int psutil_gather_unix(int proto, PyObject *py_retlist) {
static int
psutil_gather_unix(int proto, PyObject *py_retlist) {
struct xunpgen *xug, *exug;
struct xunpcb *xup;
const char *varname = NULL;
Expand Down Expand Up @@ -339,23 +349,74 @@ int psutil_gather_unix(int proto, PyObject *py_retlist) {
}


static int
psutil_int_in_seq(int value, PyObject *py_seq) {
int inseq;
PyObject *py_value;

py_value = PyLong_FromLong((long)value);
if (py_value == NULL)
return -1;
inseq = PySequence_Contains(py_seq, py_value); // return -1 on failure
Py_DECREF(py_value);
return inseq;
}


PyObject*
psutil_net_connections(PyObject* self, PyObject* args) {
// Return system-wide open connections.
int include_v4, include_v6, include_unix, include_tcp, include_udp;
PyObject *py_af_filter = NULL;
PyObject *py_type_filter = NULL;
PyObject *py_retlist = PyList_New(0);

if (py_retlist == NULL)
return NULL;
if (psutil_populate_xfiles() != 1)
if (! PyArg_ParseTuple(args, "OO", &py_af_filter, &py_type_filter)) {
goto error;
}
if (!PySequence_Check(py_af_filter) || !PySequence_Check(py_type_filter)) {
PyErr_SetString(PyExc_TypeError, "arg 2 or 3 is not a sequence");
goto error;
}

if ((include_v4 = psutil_int_in_seq(AF_INET, py_af_filter)) == -1)
goto error;
if (psutil_gather_inet(IPPROTO_TCP, py_retlist) == 0)
if ((include_v6 = psutil_int_in_seq(AF_INET6, py_af_filter)) == -1)
goto error;
if (psutil_gather_inet(IPPROTO_UDP, py_retlist) == 0)
if ((include_unix = psutil_int_in_seq(AF_UNIX, py_af_filter)) == -1)
goto error;
if (psutil_gather_unix(SOCK_STREAM, py_retlist) == 0)
goto error;
if (psutil_gather_unix(SOCK_DGRAM, py_retlist) == 0)
if ((include_tcp = psutil_int_in_seq(SOCK_STREAM, py_type_filter)) == -1)
goto error;
if ((include_udp = psutil_int_in_seq(SOCK_DGRAM, py_type_filter)) == -1)
goto error;

if (psutil_populate_xfiles() != 1)
goto error;

// TCP
if (include_tcp == 1) {
if (psutil_gather_inet(
IPPROTO_TCP, include_v4, include_v6, py_retlist) == 0)
{
goto error;
}
}
// UDP
if (include_udp == 1) {
if (psutil_gather_inet(
IPPROTO_UDP, include_v4, include_v6, py_retlist) == 0)
{
goto error;
}
}
// UNIX
if (include_unix == 1) {
if (psutil_gather_unix(SOCK_STREAM, py_retlist) == 0)
goto error;
if (psutil_gather_unix(SOCK_DGRAM, py_retlist) == 0)
goto error;
}

free(psutil_xfiles);
return py_retlist;
Expand Down

0 comments on commit 13e1fe7

Please sign in to comment.