From bc991ea8e5bc7d298a5ef9c8beeaf21d2c7418a5 Mon Sep 17 00:00:00 2001 From: Chia-Yu Chang Date: Sat, 5 Oct 2024 18:59:11 +0200 Subject: [PATCH] ss: Output TCP Prague diag information Output TCP Prague information from a struct tcp_prague_info grabbed using the inet_diag API. The Prague output includes alpha, max_burst, round, rate, cwnd if Prague is enabled; otherwise, prague:reno-fallback-mode is displayed. Co-developed-by: Olivier Tilmans Signed-off-by: Olivier Tilmans Signed-off-by: Chia-Yu Chang --- include/uapi/linux/inet_diag.h | 11 ++++++++++ misc/ss.c | 39 ++++++++++++++++++++++++++++++++++ 2 files changed, 50 insertions(+) diff --git a/include/uapi/linux/inet_diag.h b/include/uapi/linux/inet_diag.h index 2b3c82271..297c24bbc 100644 --- a/include/uapi/linux/inet_diag.h +++ b/include/uapi/linux/inet_diag.h @@ -161,6 +161,7 @@ enum { INET_DIAG_SK_BPF_STORAGES, INET_DIAG_CGROUP_ID, INET_DIAG_SOCKOPT, + INET_DIAG_PRAGUEINFO, __INET_DIAG_MAX, }; @@ -231,9 +232,19 @@ struct tcp_bbr_info { __u32 bbr_cwnd_gain; /* cwnd gain shifted left 8 bits */ }; +struct tcp_prague_info { + __u64 prague_alpha; + __u32 prague_max_burst; + __u32 prague_round; + __u64 prague_rate_bytes; + __u64 prague_frac_cwnd; + bool prague_enabled; +}; + union tcp_cc_info { struct tcpvegas_info vegas; struct tcp_dctcp_info dctcp; struct tcp_bbr_info bbr; + struct tcp_prague_info prague; }; #endif /* _INET_DIAG_H_ */ diff --git a/misc/ss.c b/misc/ss.c index aef1a714b..dce67741d 100644 --- a/misc/ss.c +++ b/misc/ss.c @@ -832,6 +832,15 @@ struct dctcpstat { bool enabled; }; +struct praguestat { + uint64_t alpha; + uint32_t max_burst; + uint32_t round; + uint64_t rate_bytes; + uint64_t frac_cwnd; + bool enabled; +}; + struct tcpstat { struct sockstat ss; unsigned int timer; @@ -895,6 +904,7 @@ struct tcpstat { bool app_limited; struct dctcpstat *dctcp; struct tcp_bbr_info *bbr_info; + struct praguestat *prague; }; /* SCTP assocs share the same inode number with their parent endpoint. So if we @@ -2671,6 +2681,18 @@ static void tcp_stats_print(struct tcpstat *s) out(" dctcp:fallback_mode"); } + if (s->prague && s->prague->enabled) { + struct praguestat *prague = s->prague; + + out(" prague:(alpha:%g%%,max_burst:%u,round:%u,rate:%lu,cwnd:%lu)", + (double)prague->alpha / (double)(1ULL << 20U) * 100.0f, + prague->max_burst, prague->round, + prague->rate_bytes, prague->frac_cwnd + ); + } else if (s->prague) { + out(" prague:reno-fallback-mode"); + } + if (s->bbr_info) { __u64 bw; @@ -3166,6 +3188,22 @@ static void tcp_show_info(const struct nlmsghdr *nlh, struct inet_diag_msg *r, s.dctcp = dctcp; } + if (tb[INET_DIAG_PRAGUEINFO]) { + struct praguestat *prague = malloc(sizeof(struct + praguestat)); + + const struct tcp_prague_info *pinfo + = RTA_DATA(tb[INET_DIAG_PRAGUEINFO]); + + prague->enabled = !!pinfo->prague_enabled; + prague->alpha = pinfo->prague_alpha; + prague->max_burst = pinfo->prague_max_burst; + prague->round = pinfo->prague_round; + prague->rate_bytes = pinfo->prague_rate_bytes; + prague->frac_cwnd = pinfo->prague_frac_cwnd; + s.prague = prague; + } + if (tb[INET_DIAG_BBRINFO]) { const void *bbr_info = RTA_DATA(tb[INET_DIAG_BBRINFO]); int len = min(RTA_PAYLOAD(tb[INET_DIAG_BBRINFO]), @@ -3216,6 +3254,7 @@ static void tcp_show_info(const struct nlmsghdr *nlh, struct inet_diag_msg *r, tcp_stats_print(&s); free(s.dctcp); free(s.bbr_info); + free(s.prague); } if (tb[INET_DIAG_MD5SIG]) { struct tcp_diag_md5sig *sig = RTA_DATA(tb[INET_DIAG_MD5SIG]);