[PATCH 1/2] rtnl_route: decode rtmsg netlink attributes

JingPiao Chen chenjingpiao at gmail.com
Wed Aug 23 00:45:18 UTC 2017


* configure.ac (AC_CHECK_TYPES): Check for
rta_mfc_stats and rtvia structures in <linux/rtnetlink.h>.
* nlattr.c (decode_nla_ifindex): New function.
* nlattr.h (decode_nla_ifindex): New prototype.
* rtnl_route.c (decode_route_addr, decode_rta_metrics,
decode_rta_multipath, decode_rta_cacheinfo,
decode_rta_mfc_stats, decode_rtvia,
decode_rta_encap_type): New functions.
(rta_metrics_nla_decoders, rtmsg_nla_decoders): New arrays.
(decode_rtmsg): Use rtmsg_nla_decoders.
* xlat/lwtunnel_encap_types.in: New file.
* xlat/route_nexthop_flags.in: Likewise.
* xlat/rtnl_rta_metrics_attrs.in: Likewise.
---
 configure.ac                   |   5 +
 nlattr.c                       |  16 +++
 nlattr.h                       |   1 +
 rtnl_route.c                   | 215 ++++++++++++++++++++++++++++++++++++++++-
 xlat/lwtunnel_encap_types.in   |   7 ++
 xlat/route_nexthop_flags.in    |   6 ++
 xlat/rtnl_rta_metrics_attrs.in |  17 ++++
 7 files changed, 266 insertions(+), 1 deletion(-)
 create mode 100644 xlat/lwtunnel_encap_types.in
 create mode 100644 xlat/route_nexthop_flags.in
 create mode 100644 xlat/rtnl_rta_metrics_attrs.in

diff --git a/configure.ac b/configure.ac
index b4abdf8..4522b2e 100644
--- a/configure.ac
+++ b/configure.ac
@@ -447,6 +447,11 @@ AC_CHECK_TYPES([struct dcbmsg],,, [#include <linux/dcbnl.h>])
 AC_CHECK_TYPES([struct ifaddrlblmsg],,, [#include <linux/if_addrlabel.h>])
 AC_CHECK_TYPES([struct netconfmsg],,, [#include <linux/netconf.h>])
 
+AC_CHECK_TYPES(m4_normalize([
+	struct rta_mfc_stats,
+	struct rtvia
+]),,, [#include <linux/rtnetlink.h>])
+
 AC_CHECK_TYPES([struct statfs], [
 	AC_CHECK_MEMBERS(m4_normalize([
 		struct statfs.f_frsize,
diff --git a/nlattr.c b/nlattr.c
index b0786df..76010c4 100644
--- a/nlattr.c
+++ b/nlattr.c
@@ -204,6 +204,22 @@ decode_nla_meminfo(struct tcb *const tcp,
 	return true;
 }
 
+bool
+decode_nla_ifindex(struct tcb *const tcp,
+	       const kernel_ulong_t addr,
+	       const unsigned int len,
+	       const void *const opaque_data)
+{
+	uint32_t ifindex;
+
+	if (len < sizeof(ifindex))
+		return false;
+	else if (!umove_or_printaddr(tcp, addr, &ifindex))
+		print_ifindex(ifindex);
+
+	return true;
+}
+
 #define DECODE_NLA_INTEGER(name, type, fmt)		\
 bool							\
 decode_nla_ ## name(struct tcb *const tcp,		\
diff --git a/nlattr.h b/nlattr.h
index 9437883..636c787 100644
--- a/nlattr.h
+++ b/nlattr.h
@@ -58,6 +58,7 @@ DECL_NLA(s32);
 DECL_NLA(s64);
 DECL_NLA(str);
 DECL_NLA(strn);
+DECL_NLA(ifindex);
 DECL_NLA(meminfo);
 
 #endif /* !STRACE_NLATTR_H */
diff --git a/rtnl_route.c b/rtnl_route.c
index 7b997f8..8bb9ba9 100644
--- a/rtnl_route.c
+++ b/rtnl_route.c
@@ -37,11 +37,222 @@
 #include <linux/rtnetlink.h>
 
 #include "xlat/ip_type_of_services.h"
+#include "xlat/lwtunnel_encap_types.h"
+#include "xlat/route_nexthop_flags.h"
 #include "xlat/routing_flags.h"
 #include "xlat/routing_protocols.h"
 #include "xlat/routing_table_ids.h"
 #include "xlat/routing_types.h"
 #include "xlat/rtnl_route_attrs.h"
+#include "xlat/rtnl_rta_metrics_attrs.h"
+
+static bool
+decode_route_addr(struct tcb *const tcp,
+		  const kernel_ulong_t addr,
+		  const unsigned int len,
+		  const void *const opaque_data)
+{
+	const struct rtmsg *const rtmsg = opaque_data;
+
+	decode_inet_addr(tcp, addr, len, rtmsg->rtm_family, NULL);
+
+	return true;
+}
+
+static const nla_decoder_t rta_metrics_nla_decoders[] = {
+	[RTAX_LOCK]		= decode_nla_u32,
+	[RTAX_MTU]		= decode_nla_u32,
+	[RTAX_WINDOW]		= decode_nla_u32,
+	[RTAX_RTT]		= decode_nla_u32,
+	[RTAX_RTTVAR]		= decode_nla_u32,
+	[RTAX_SSTHRESH]		= decode_nla_u32,
+	[RTAX_CWND]		= decode_nla_u32,
+	[RTAX_ADVMSS]		= decode_nla_u32,
+	[RTAX_REORDERING]	= decode_nla_u32,
+	[RTAX_HOPLIMIT]		= decode_nla_u32,
+	[RTAX_INITCWND]		= decode_nla_u32,
+	[RTAX_FEATURES]		= decode_nla_u32,
+	[RTAX_RTO_MIN]		= decode_nla_u32,
+	[RTAX_INITRWND]		= decode_nla_u32,
+	[RTAX_QUICKACK]		= decode_nla_u32,
+	[RTAX_CC_ALGO]		= decode_nla_str
+};
+
+static bool
+decode_rta_metrics(struct tcb *const tcp,
+		   const kernel_ulong_t addr,
+		   const unsigned int len,
+		   const void *const opaque_data)
+{
+	decode_nlattr(tcp, addr, len, rtnl_rta_metrics_attrs,
+		      "RTAX_???", rta_metrics_nla_decoders,
+		      ARRAY_SIZE(rta_metrics_nla_decoders), opaque_data);
+
+	return true;
+}
+
+static bool
+decode_rta_multipath(struct tcb *const tcp,
+		     const kernel_ulong_t addr,
+		     const unsigned int len,
+		     const void *const opaque_data);
+
+static bool
+decode_rta_cacheinfo(struct tcb *const tcp,
+		     const kernel_ulong_t addr,
+		     const unsigned int len,
+		     const void *const opaque_data)
+{
+	struct rta_cacheinfo ci;
+
+	if (len < sizeof(ci))
+		return false;
+	else if (!umove_or_printaddr(tcp, addr, &ci)) {
+		PRINT_FIELD_U("{", ci, rta_clntref);
+		PRINT_FIELD_U(", ", ci, rta_lastuse);
+		PRINT_FIELD_U(", ", ci, rta_expires);
+		PRINT_FIELD_U(", ", ci, rta_error);
+		PRINT_FIELD_U(", ", ci, rta_used);
+		PRINT_FIELD_X(", ", ci, rta_id);
+		PRINT_FIELD_U(", ", ci, rta_ts);
+		PRINT_FIELD_U(", ", ci, rta_tsage);
+		tprints("}");
+	}
+
+	return true;
+}
+
+static bool
+decode_rta_mfc_stats(struct tcb *const tcp,
+		     const kernel_ulong_t addr,
+		     const unsigned int len,
+		     const void *const opaque_data)
+{
+#ifdef HAVE_STRUCT_RTA_MFC_STATS
+	struct rta_mfc_stats mfcs;
+
+	if (len < sizeof(mfcs))
+		return false;
+	else if (!umove_or_printaddr(tcp, addr, &mfcs)) {
+		PRINT_FIELD_U("{", mfcs, mfcs_packets);
+		PRINT_FIELD_U(", ", mfcs, mfcs_bytes);
+		PRINT_FIELD_U(", ", mfcs, mfcs_wrong_if);
+		tprints("}");
+	}
+
+	return true;
+#else
+	return false;
+#endif
+}
+
+static bool
+decode_rtvia(struct tcb *const tcp,
+	     const kernel_ulong_t addr,
+	     const unsigned int len,
+	     const void *const opaque_data)
+{
+#ifdef HAVE_STRUCT_RTVIA
+	struct rtvia via;
+
+	if (len < sizeof(via))
+		return false;
+	else if (!umove_or_printaddr(tcp, addr, &via)) {
+		PRINT_FIELD_XVAL("{", via, rtvia_family, addrfams, "AF_???");
+
+		if (len > sizeof(via)) {
+			tprints(", rtvia_addr=");
+			printstr_ex(tcp, addr + sizeof(via),
+				    len - sizeof(via), QUOTE_FORCE_HEX);
+		}
+		tprints("}");
+	}
+
+	return true;
+#else
+	return false;
+#endif
+}
+
+static bool
+decode_rta_encap_type(struct tcb *const tcp,
+		      const kernel_ulong_t addr,
+		      const unsigned int len,
+		      const void *const opaque_data)
+{
+	uint16_t type;
+
+	if (len < sizeof(type))
+		return false;
+	else if (!umove_or_printaddr(tcp, addr, &type))
+		printxval(lwtunnel_encap_types, type, "LWTUNNEL_ENCAP_???");
+
+	return true;
+}
+
+static const nla_decoder_t rtmsg_nla_decoders[] = {
+	[RTA_DST]		= decode_route_addr,
+	[RTA_SRC]		= decode_route_addr,
+	[RTA_IIF]		= decode_nla_ifindex,
+	[RTA_OIF]		= decode_nla_ifindex,
+	[RTA_GATEWAY]		= decode_route_addr,
+	[RTA_PRIORITY]		= decode_nla_u32,
+	[RTA_PREFSRC]		= decode_route_addr,
+	[RTA_METRICS]		= decode_rta_metrics,
+	[RTA_MULTIPATH]		= decode_rta_multipath,
+	[RTA_PROTOINFO]		= decode_nla_u32,
+	[RTA_FLOW]		= decode_nla_u32,
+	[RTA_CACHEINFO]		= decode_rta_cacheinfo,
+	[RTA_SESSION]		= NULL, /* unused */
+	[RTA_MP_ALGO]		= decode_nla_u32,
+	[RTA_TABLE]		= decode_nla_u32,
+	[RTA_MARK]		= decode_nla_u32,
+	[RTA_MFC_STATS]		= decode_rta_mfc_stats,
+	[RTA_VIA]		= decode_rtvia,
+	[RTA_NEWDST]		= decode_route_addr,
+	[RTA_PREF]		= decode_nla_u8,
+	[RTA_ENCAP_TYPE]	= decode_rta_encap_type,
+	[RTA_ENCAP]		= NULL, /* unimplemented */
+	[RTA_EXPIRES]		= decode_nla_u64,
+	[RTA_PAD]		= NULL,
+	[RTA_UID]		= decode_nla_u32,
+	[RTA_TTL_PROPAGATE]	= decode_nla_u8
+};
+
+static bool
+decode_rta_multipath(struct tcb *const tcp,
+		     const kernel_ulong_t addr,
+		     const unsigned int len,
+		     const void *const opaque_data)
+{
+	struct rtnexthop nh;
+
+	if (len < sizeof(nh))
+		return false;
+	else if (!umove_or_printaddr(tcp, addr, &nh)) {
+		/* print the whole structure regardless of its rtnh_len */
+		PRINT_FIELD_U("{", nh, rtnh_len);
+		PRINT_FIELD_FLAGS(", ", nh, rtnh_flags,
+				  route_nexthop_flags, "RTNH_F_???");
+		PRINT_FIELD_U(", ", nh, rtnh_hops);
+		PRINT_FIELD_IFINDEX(", ", nh, rtnh_ifindex);
+		tprints("}");
+
+		const unsigned short rtnh_len =
+			len < nh.rtnh_len ? len : nh.rtnh_len;
+		const size_t offset = RTNH_ALIGN(sizeof(nh));
+		if (rtnh_len > offset) {
+			tprints(", ");
+			decode_nlattr(tcp, addr + offset, rtnh_len - offset,
+				      rtnl_route_attrs, "RTA_???",
+				      rtmsg_nla_decoders,
+				      ARRAY_SIZE(rtmsg_nla_decoders),
+				      opaque_data);
+		}
+	}
+
+	return true;
+}
 
 DECL_NETLINK_ROUTE_DECODER(decode_rtmsg)
 {
@@ -80,6 +291,8 @@ DECL_NETLINK_ROUTE_DECODER(decode_rtmsg)
 	if (decode_nla && len > offset) {
 		tprints(", ");
 		decode_nlattr(tcp, addr + offset, len - offset,
-			      rtnl_route_attrs, "RTA_???", NULL, 0, NULL);
+			      rtnl_route_attrs, "RTA_???",
+			      rtmsg_nla_decoders,
+			      ARRAY_SIZE(rtmsg_nla_decoders), &rtmsg);
 	}
 }
diff --git a/xlat/lwtunnel_encap_types.in b/xlat/lwtunnel_encap_types.in
new file mode 100644
index 0000000..4654bd4
--- /dev/null
+++ b/xlat/lwtunnel_encap_types.in
@@ -0,0 +1,7 @@
+LWTUNNEL_ENCAP_NONE	0
+LWTUNNEL_ENCAP_MPLS	1
+LWTUNNEL_ENCAP_IP	2
+LWTUNNEL_ENCAP_ILA	3
+LWTUNNEL_ENCAP_IP6	4
+LWTUNNEL_ENCAP_SEG6	5
+LWTUNNEL_ENCAP_BPF	6
diff --git a/xlat/route_nexthop_flags.in b/xlat/route_nexthop_flags.in
new file mode 100644
index 0000000..907515e
--- /dev/null
+++ b/xlat/route_nexthop_flags.in
@@ -0,0 +1,6 @@
+RTNH_F_DEAD
+RTNH_F_PERVASIVE
+RTNH_F_ONLINK
+RTNH_F_OFFLOAD
+RTNH_F_LINKDOWN
+RTNH_F_UNRESOLVED
diff --git a/xlat/rtnl_rta_metrics_attrs.in b/xlat/rtnl_rta_metrics_attrs.in
new file mode 100644
index 0000000..2a68333
--- /dev/null
+++ b/xlat/rtnl_rta_metrics_attrs.in
@@ -0,0 +1,17 @@
+RTAX_UNSPEC	0
+RTAX_LOCK	1
+RTAX_MTU	2
+RTAX_WINDOW	3
+RTAX_RTT	4
+RTAX_RTTVAR	5
+RTAX_SSTHRESH	6
+RTAX_CWND	7
+RTAX_ADVMSS	8
+RTAX_REORDERING	9
+RTAX_HOPLIMIT	10
+RTAX_INITCWND	11
+RTAX_FEATURES	12
+RTAX_RTO_MIN	13
+RTAX_INITRWND	14
+RTAX_QUICKACK	15
+RTAX_CC_ALGO	16
-- 
2.7.4





More information about the Strace-devel mailing list