You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

280 lines
7.6 KiB

/* SPDX-License-Identifier: LGPL-2.1-only */
/*
* lib/idiag/idiag.c Inet Diag Netlink
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation version 2.1
* of the License.
*
* Copyright (c) 2013 Sassano Systems LLC <joe@sassanosystems.com>
*/
/**
* @defgroup idiag Inet Diag library (libnl-idiag)
* @brief
* @{
*/
#include <netlink-private/netlink.h>
#include <netlink/netlink.h>
#include <netlink/cache.h>
#include <netlink/idiag/idiagnl.h>
#include <linux/inet_diag.h>
/**
* @name Socket Creation
* @{
*/
/**
* Create and connect idiag netlink socket.
* @arg sk Netlink socket.
*
* Creates a NETLINK_INET_DIAG socket, binds the socket, and issues a connection
* attemp.
*
* @see nl_connect()
*
* @return 0 on success or a negative error code.
*/
int idiagnl_connect(struct nl_sock *sk)
{
return nl_connect(sk, NETLINK_INET_DIAG);
}
/** @} */
/**
* @name Sending
* @{
*/
/**
* Send trivial idiag netlink message
* @arg sk Netlink socket.
* @arg flags Message flags
* @arg family Address family
* @arg states Socket states to query
* @arg ext Inet Diag attribute extensions to query. Note that this only supports
* 8 bit arguments. Flags outside uint8_t range are silently ignored.
*
* @return 0 on success or a negative error code. Due to a bug, this function
* returns the number of bytes sent. Treat any non-negative number as success.
*/
int idiagnl_send_simple(struct nl_sock *sk, int flags, uint8_t family,
uint16_t states, uint16_t ext)
{
struct inet_diag_req req;
memset(&req, 0, sizeof(req));
flags |= NLM_F_ROOT;
req.idiag_family = family;
req.idiag_states = states;
req.idiag_ext = ext;
return nl_send_simple(sk, TCPDIAG_GETSOCK, flags, &req, sizeof(req));
}
/** @} */
/**
* @name Inet Diag flag and attribute conversions
* @{
*/
static const struct trans_tbl idiag_states[] = {
__ADD(TCP_ESTABLISHED, established),
__ADD(TCP_SYN_SENT, syn_sent),
__ADD(TCP_SYN_RECV, syn_recv),
__ADD(TCP_FIN_WAIT1, fin_wait),
__ADD(TCP_FIN_WAIT2, fin_wait2),
__ADD(TCP_TIME_WAIT, time_wait),
__ADD(TCP_CLOSE, close),
__ADD(TCP_CLOSE_WAIT, close_wait),
__ADD(TCP_LAST_ACK, last_ack),
__ADD(TCP_LISTEN, listen),
__ADD(TCP_CLOSING, closing),
};
/**
* Convert inet diag socket states to strings.
* @arg state inetdiag socket state (e.g., TCP_ESTABLISHED)
* @arg buf output buffer which will hold string result
* @arg len length in bytes of the output buffer
*
* @return string representation of the inetdiag socket state or an empty
* string.
*/
char * idiagnl_state2str(int state, char *buf, size_t len)
{
return __type2str(state, buf, len, idiag_states,
ARRAY_SIZE(idiag_states));
}
/**
* Convert inet diag socket state string to int.
* @arg name inetdiag socket state string
*
* @return the int representation of the socket state strign or a negative error
* code.
*/
int idiagnl_str2state(const char *name)
{
return __str2type(name, idiag_states, ARRAY_SIZE(idiag_states));
}
static const struct trans_tbl idiag_timers[] = {
__ADD(IDIAGNL_TIMER_OFF, off),
__ADD(IDIAGNL_TIMER_ON, on),
__ADD(IDIAGNL_TIMER_KEEPALIVE, keepalive),
__ADD(IDIAGNL_TIMER_TIMEWAIT, timewait),
__ADD(IDIAGNL_TIMER_PERSIST, persist),
__ADD(IDIAGNL_TIMER_UNKNOWN, unknown),
};
/**
* Convert inet diag timer types to strings.
* @arg timer inetdiag timer (e.g., IDIAGNL_TIMER_ON)
* @arg buf output buffer which will hold string result
* @arg len length in bytes of the output buffer
*
* @return string representation of the inetdiag timer type or an empty string.
*/
char * idiagnl_timer2str(int timer, char *buf, size_t len)
{
return __type2str(timer, buf, len, idiag_timers,
ARRAY_SIZE(idiag_timers));
}
/**
* Convert inet diag timer string to int.
* @arg name inetdiag timer string
*
* @return the int representation of the timer string or a negative error code.
*/
int idiagnl_str2timer(const char *name)
{
return __str2type(name, idiag_timers, ARRAY_SIZE(idiag_timers));
}
static const struct trans_tbl idiag_attrs[] = {
__ADD(INET_DIAG_NONE, none),
__ADD(INET_DIAG_MEMINFO, meminfo),
__ADD(INET_DIAG_INFO, info),
__ADD(INET_DIAG_VEGASINFO, vegasinfo),
__ADD(INET_DIAG_CONG, congestion),
__ADD(INET_DIAG_TOS, tos),
__ADD(INET_DIAG_TCLASS, tclass),
__ADD(INET_DIAG_SKMEMINFO, skmeminfo),
__ADD(INET_DIAG_SHUTDOWN, shutdown),
};
/**
* Convert inet diag extension type to a string.
* @arg attrs inet diag extension type (e.g. INET_DIAG_MEMINFO)
* @arg buf output buffer which will hold string result
* @arg len length in bytes of the output buffer
*
* @return string representation of inet diag extension type or an empty string.
* @deprecated: don't use this function. It is not very useful and should
* never have been exposed as public API.
*/
char *idiagnl_attrs2str(int attrs, char *buf, size_t len)
{
return __type2str(attrs, buf, len, idiag_attrs, ARRAY_SIZE(idiag_attrs));
}
static const struct trans_tbl idiag_exts[] = {
__ADD((1 << (INET_DIAG_MEMINFO - 1)), meminfo),
__ADD((1 << (INET_DIAG_INFO - 1)), info),
__ADD((1 << (INET_DIAG_VEGASINFO - 1)), vegasinfo),
__ADD((1 << (INET_DIAG_CONG - 1)), congestion),
__ADD((1 << (INET_DIAG_TOS - 1)), tos),
__ADD((1 << (INET_DIAG_TCLASS - 1)), tclass),
__ADD((1 << (INET_DIAG_SKMEMINFO - 1)), skmeminfo),
__ADD((1 << (INET_DIAG_SHUTDOWN - 1)), shutdown),
};
/**
* Convert inet diag extension flags to a string.
* @arg attrs inet diag extension flags (e.g.
* ( (1<<(INET_DIAG_MEMINFO-1)) | (1<<(INET_DIAG_CONG-1)) | (1<<(INET_DIAG_TOS-1)) ) )
* @arg buf Output buffer to hold string representation
* @arg len length in bytes of the output buffer
*/
char *idiagnl_exts2str(uint8_t attrs, char *buf, size_t len)
{
return __flags2str(attrs, buf, len, idiag_exts, ARRAY_SIZE(idiag_exts));
}
static const struct trans_tbl idiagnl_tcpstates[] = {
__ADD(TCP_CA_Open, open),
__ADD(TCP_CA_Disorder, disorder),
__ADD(TCP_CA_CWR, cwr),
__ADD(TCP_CA_Recovery, recovery),
__ADD(TCP_CA_Loss, loss),
};
/**
* Convert inetdiag tcp states to strings.
* @arg state TCP state (e.g., TCP_CA_Open)
* @arg buf output buffer which will hold string result
* @arg len length in bytes of the output buffer
*/
char *idiagnl_tcpstate2str(uint8_t state, char *buf, size_t len)
{
return __type2str(state, buf, len, idiagnl_tcpstates,
ARRAY_SIZE(idiagnl_tcpstates));
}
static const struct trans_tbl idiagnl_tcpopt_attrs[] = {
__ADD(TCPI_OPT_TIMESTAMPS, timestamps),
__ADD(TCPI_OPT_SACK, sACK),
__ADD(TCPI_OPT_WSCALE, wscale),
__ADD(TCPI_OPT_ECN, ecn),
};
/**
* Convert TCP option attributes to string
* @arg attrs TCP option attributes to convert (e.g., TCPI_OPT_SACK |
* TCPI_OPT_WSCALE)
* @arg buf Output buffer for string
* @arg len Length in bytes of output buffer
*
* @return buffer with string representation or empty string
*/
char *idiagnl_tcpopts2str(uint8_t attrs, char *buf, size_t len)
{
return __flags2str(attrs, buf, len, idiagnl_tcpopt_attrs,
ARRAY_SIZE(idiagnl_tcpopt_attrs));
}
/**
* Convert shutdown state to string.
* @arg shutdown Shutdown state (e.g., idiag_msg->shutdown)
* @arg buf Ouput buffer to hold string representation
* @arg len Length in bytes of output buffer
*
* @return string representation of shutdown state or NULL
*/
char * idiagnl_shutdown2str(uint8_t shutdown, char *buf, size_t len)
{
if (shutdown == 0) {
snprintf(buf, len, " ");
return buf;
} else if (shutdown == 1) {
snprintf(buf, len, "receive shutdown");
return buf;
} else if (shutdown == 2) {
snprintf(buf, len, "send shutdown");
return buf;
}
return NULL;
}
/** @} */
/** @} */