tools: ynl: generate code for the ethtool family
authorJakub Kicinski <kuba@kernel.org>
Fri, 9 Jun 2023 21:43:45 +0000 (14:43 -0700)
committerDavid S. Miller <davem@davemloft.net>
Mon, 12 Jun 2023 10:01:03 +0000 (11:01 +0100)
Generate the protocol code for ethtool. Skip the stats
for now, they are the only outlier in terms of complexity.
Stats are a sort-of semi-polymorphic (attr space of a nest
depends on value of another attr) or a type-value-scalar,
depending on how one wants to look at it...
A challenge for another time.

Signed-off-by: Jakub Kicinski <kuba@kernel.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
tools/net/ynl/generated/Makefile
tools/net/ynl/generated/ethtool-user.c [new file with mode: 0644]
tools/net/ynl/generated/ethtool-user.h [new file with mode: 0644]

index 8ce6109..f15c248 100644 (file)
@@ -7,9 +7,12 @@ ifeq ("$(DEBUG)","1")
   CFLAGS += -g -fsanitize=address -fsanitize=leak -static-libasan
 endif
 
+YNL_GEN_ARG_ethtool:=--user-header linux/ethtool_netlink.h \
+       --exclude-op stats-get
+
 TOOL:=../ynl-gen-c.py
 
-GENS:=devlink handshake fou netdev
+GENS:=ethtool devlink handshake fou netdev
 SRCS=$(patsubst %,%-user.c,${GENS})
 HDRS=$(patsubst %,%-user.h,${GENS})
 OBJS=$(patsubst %,%-user.o,${GENS})
@@ -22,11 +25,11 @@ protos.a: $(OBJS)
 
 %-user.h: ../../../../Documentation/netlink/specs/%.yaml $(TOOL)
        @echo -e "\tGEN $@"
-       @$(TOOL) --mode user --header --spec $< > $@
+       @$(TOOL) --mode user --header --spec $< $(YNL_GEN_ARG_$*) > $@
 
 %-user.c: ../../../../Documentation/netlink/specs/%.yaml $(TOOL)
        @echo -e "\tGEN $@"
-       @$(TOOL) --mode user --source --spec $< > $@
+       @$(TOOL) --mode user --source --spec $< $(YNL_GEN_ARG_$*) > $@
 
 %-user.o: %-user.c %-user.h
        @echo -e "\tCC $@"
diff --git a/tools/net/ynl/generated/ethtool-user.c b/tools/net/ynl/generated/ethtool-user.c
new file mode 100644 (file)
index 0000000..74b883a
--- /dev/null
@@ -0,0 +1,6353 @@
+// SPDX-License-Identifier: ((GPL-2.0 WITH Linux-syscall-note) OR BSD-3-Clause)
+/* Do not edit directly, auto-generated from: */
+/*     Documentation/netlink/specs/ethtool.yaml */
+/* YNL-GEN user source */
+/* YNL-ARG --user-header linux/ethtool_netlink.h --exclude-op stats-get */
+
+#include <stdlib.h>
+#include <string.h>
+#include "ethtool-user.h"
+#include "ynl.h"
+#include <linux/ethtool.h>
+
+#include <libmnl/libmnl.h>
+#include <linux/genetlink.h>
+
+#include "linux/ethtool_netlink.h"
+
+/* Enums */
+static const char * const ethtool_op_strmap[] = {
+       [ETHTOOL_MSG_STRSET_GET] = "strset-get",
+       [ETHTOOL_MSG_LINKINFO_GET] = "linkinfo-get",
+       [3] = "linkinfo-ntf",
+       [ETHTOOL_MSG_LINKMODES_GET] = "linkmodes-get",
+       [5] = "linkmodes-ntf",
+       [ETHTOOL_MSG_LINKSTATE_GET] = "linkstate-get",
+       [ETHTOOL_MSG_DEBUG_GET] = "debug-get",
+       [8] = "debug-ntf",
+       [ETHTOOL_MSG_WOL_GET] = "wol-get",
+       [10] = "wol-ntf",
+       [ETHTOOL_MSG_FEATURES_GET] = "features-get",
+       [ETHTOOL_MSG_FEATURES_SET] = "features-set",
+       [13] = "features-ntf",
+       [14] = "privflags-get",
+       [15] = "privflags-ntf",
+       [16] = "rings-get",
+       [17] = "rings-ntf",
+       [18] = "channels-get",
+       [19] = "channels-ntf",
+       [20] = "coalesce-get",
+       [21] = "coalesce-ntf",
+       [22] = "pause-get",
+       [23] = "pause-ntf",
+       [24] = "eee-get",
+       [25] = "eee-ntf",
+       [26] = "tsinfo-get",
+       [27] = "cable-test-ntf",
+       [28] = "cable-test-tdr-ntf",
+       [29] = "tunnel-info-get",
+       [30] = "fec-get",
+       [31] = "fec-ntf",
+       [32] = "module-eeprom-get",
+       [34] = "phc-vclocks-get",
+       [35] = "module-get",
+       [36] = "module-ntf",
+       [37] = "pse-get",
+       [ETHTOOL_MSG_RSS_GET] = "rss-get",
+       [ETHTOOL_MSG_PLCA_GET_CFG] = "plca-get-cfg",
+       [40] = "plca-get-status",
+       [41] = "plca-ntf",
+       [ETHTOOL_MSG_MM_GET] = "mm-get",
+       [43] = "mm-ntf",
+};
+
+const char *ethtool_op_str(int op)
+{
+       if (op < 0 || op >= (int)MNL_ARRAY_SIZE(ethtool_op_strmap))
+               return NULL;
+       return ethtool_op_strmap[op];
+}
+
+static const char * const ethtool_udp_tunnel_type_strmap[] = {
+       [0] = "vxlan",
+       [1] = "geneve",
+       [2] = "vxlan-gpe",
+};
+
+const char *ethtool_udp_tunnel_type_str(int value)
+{
+       if (value < 0 || value >= (int)MNL_ARRAY_SIZE(ethtool_udp_tunnel_type_strmap))
+               return NULL;
+       return ethtool_udp_tunnel_type_strmap[value];
+}
+
+static const char * const ethtool_stringset_strmap[] = {
+};
+
+const char *ethtool_stringset_str(enum ethtool_stringset value)
+{
+       if (value < 0 || value >= (int)MNL_ARRAY_SIZE(ethtool_stringset_strmap))
+               return NULL;
+       return ethtool_stringset_strmap[value];
+}
+
+/* Policies */
+struct ynl_policy_attr ethtool_header_policy[ETHTOOL_A_HEADER_MAX + 1] = {
+       [ETHTOOL_A_HEADER_DEV_INDEX] = { .name = "dev-index", .type = YNL_PT_U32, },
+       [ETHTOOL_A_HEADER_DEV_NAME] = { .name = "dev-name", .type = YNL_PT_NUL_STR, },
+       [ETHTOOL_A_HEADER_FLAGS] = { .name = "flags", .type = YNL_PT_U32, },
+};
+
+struct ynl_policy_nest ethtool_header_nest = {
+       .max_attr = ETHTOOL_A_HEADER_MAX,
+       .table = ethtool_header_policy,
+};
+
+struct ynl_policy_attr ethtool_pause_stat_policy[ETHTOOL_A_PAUSE_STAT_MAX + 1] = {
+       [ETHTOOL_A_PAUSE_STAT_PAD] = { .name = "pad", .type = YNL_PT_IGNORE, },
+       [ETHTOOL_A_PAUSE_STAT_TX_FRAMES] = { .name = "tx-frames", .type = YNL_PT_U64, },
+       [ETHTOOL_A_PAUSE_STAT_RX_FRAMES] = { .name = "rx-frames", .type = YNL_PT_U64, },
+};
+
+struct ynl_policy_nest ethtool_pause_stat_nest = {
+       .max_attr = ETHTOOL_A_PAUSE_STAT_MAX,
+       .table = ethtool_pause_stat_policy,
+};
+
+struct ynl_policy_attr ethtool_cable_test_tdr_cfg_policy[ETHTOOL_A_CABLE_TEST_TDR_CFG_MAX + 1] = {
+       [ETHTOOL_A_CABLE_TEST_TDR_CFG_FIRST] = { .name = "first", .type = YNL_PT_U32, },
+       [ETHTOOL_A_CABLE_TEST_TDR_CFG_LAST] = { .name = "last", .type = YNL_PT_U32, },
+       [ETHTOOL_A_CABLE_TEST_TDR_CFG_STEP] = { .name = "step", .type = YNL_PT_U32, },
+       [ETHTOOL_A_CABLE_TEST_TDR_CFG_PAIR] = { .name = "pair", .type = YNL_PT_U8, },
+};
+
+struct ynl_policy_nest ethtool_cable_test_tdr_cfg_nest = {
+       .max_attr = ETHTOOL_A_CABLE_TEST_TDR_CFG_MAX,
+       .table = ethtool_cable_test_tdr_cfg_policy,
+};
+
+struct ynl_policy_attr ethtool_fec_stat_policy[ETHTOOL_A_FEC_STAT_MAX + 1] = {
+       [ETHTOOL_A_FEC_STAT_PAD] = { .name = "pad", .type = YNL_PT_IGNORE, },
+       [ETHTOOL_A_FEC_STAT_CORRECTED] = { .name = "corrected", .type = YNL_PT_BINARY,},
+       [ETHTOOL_A_FEC_STAT_UNCORR] = { .name = "uncorr", .type = YNL_PT_BINARY,},
+       [ETHTOOL_A_FEC_STAT_CORR_BITS] = { .name = "corr-bits", .type = YNL_PT_BINARY,},
+};
+
+struct ynl_policy_nest ethtool_fec_stat_nest = {
+       .max_attr = ETHTOOL_A_FEC_STAT_MAX,
+       .table = ethtool_fec_stat_policy,
+};
+
+struct ynl_policy_attr ethtool_mm_stat_policy[ETHTOOL_A_MM_STAT_MAX + 1] = {
+       [ETHTOOL_A_MM_STAT_PAD] = { .name = "pad", .type = YNL_PT_IGNORE, },
+       [ETHTOOL_A_MM_STAT_REASSEMBLY_ERRORS] = { .name = "reassembly-errors", .type = YNL_PT_U64, },
+       [ETHTOOL_A_MM_STAT_SMD_ERRORS] = { .name = "smd-errors", .type = YNL_PT_U64, },
+       [ETHTOOL_A_MM_STAT_REASSEMBLY_OK] = { .name = "reassembly-ok", .type = YNL_PT_U64, },
+       [ETHTOOL_A_MM_STAT_RX_FRAG_COUNT] = { .name = "rx-frag-count", .type = YNL_PT_U64, },
+       [ETHTOOL_A_MM_STAT_TX_FRAG_COUNT] = { .name = "tx-frag-count", .type = YNL_PT_U64, },
+       [ETHTOOL_A_MM_STAT_HOLD_COUNT] = { .name = "hold-count", .type = YNL_PT_U64, },
+};
+
+struct ynl_policy_nest ethtool_mm_stat_nest = {
+       .max_attr = ETHTOOL_A_MM_STAT_MAX,
+       .table = ethtool_mm_stat_policy,
+};
+
+struct ynl_policy_attr ethtool_cable_result_policy[ETHTOOL_A_CABLE_RESULT_MAX + 1] = {
+       [ETHTOOL_A_CABLE_RESULT_PAIR] = { .name = "pair", .type = YNL_PT_U8, },
+       [ETHTOOL_A_CABLE_RESULT_CODE] = { .name = "code", .type = YNL_PT_U8, },
+};
+
+struct ynl_policy_nest ethtool_cable_result_nest = {
+       .max_attr = ETHTOOL_A_CABLE_RESULT_MAX,
+       .table = ethtool_cable_result_policy,
+};
+
+struct ynl_policy_attr ethtool_cable_fault_length_policy[ETHTOOL_A_CABLE_FAULT_LENGTH_MAX + 1] = {
+       [ETHTOOL_A_CABLE_FAULT_LENGTH_PAIR] = { .name = "pair", .type = YNL_PT_U8, },
+       [ETHTOOL_A_CABLE_FAULT_LENGTH_CM] = { .name = "cm", .type = YNL_PT_U32, },
+};
+
+struct ynl_policy_nest ethtool_cable_fault_length_nest = {
+       .max_attr = ETHTOOL_A_CABLE_FAULT_LENGTH_MAX,
+       .table = ethtool_cable_fault_length_policy,
+};
+
+struct ynl_policy_attr ethtool_bitset_bit_policy[ETHTOOL_A_BITSET_BIT_MAX + 1] = {
+       [ETHTOOL_A_BITSET_BIT_INDEX] = { .name = "index", .type = YNL_PT_U32, },
+       [ETHTOOL_A_BITSET_BIT_NAME] = { .name = "name", .type = YNL_PT_NUL_STR, },
+       [ETHTOOL_A_BITSET_BIT_VALUE] = { .name = "value", .type = YNL_PT_FLAG, },
+};
+
+struct ynl_policy_nest ethtool_bitset_bit_nest = {
+       .max_attr = ETHTOOL_A_BITSET_BIT_MAX,
+       .table = ethtool_bitset_bit_policy,
+};
+
+struct ynl_policy_attr ethtool_tunnel_udp_entry_policy[ETHTOOL_A_TUNNEL_UDP_ENTRY_MAX + 1] = {
+       [ETHTOOL_A_TUNNEL_UDP_ENTRY_PORT] = { .name = "port", .type = YNL_PT_U16, },
+       [ETHTOOL_A_TUNNEL_UDP_ENTRY_TYPE] = { .name = "type", .type = YNL_PT_U32, },
+};
+
+struct ynl_policy_nest ethtool_tunnel_udp_entry_nest = {
+       .max_attr = ETHTOOL_A_TUNNEL_UDP_ENTRY_MAX,
+       .table = ethtool_tunnel_udp_entry_policy,
+};
+
+struct ynl_policy_attr ethtool_string_policy[ETHTOOL_A_STRING_MAX + 1] = {
+       [ETHTOOL_A_STRING_INDEX] = { .name = "index", .type = YNL_PT_U32, },
+       [ETHTOOL_A_STRING_VALUE] = { .name = "value", .type = YNL_PT_NUL_STR, },
+};
+
+struct ynl_policy_nest ethtool_string_nest = {
+       .max_attr = ETHTOOL_A_STRING_MAX,
+       .table = ethtool_string_policy,
+};
+
+struct ynl_policy_attr ethtool_cable_nest_policy[ETHTOOL_A_CABLE_NEST_MAX + 1] = {
+       [ETHTOOL_A_CABLE_NEST_RESULT] = { .name = "result", .type = YNL_PT_NEST, .nest = &ethtool_cable_result_nest, },
+       [ETHTOOL_A_CABLE_NEST_FAULT_LENGTH] = { .name = "fault-length", .type = YNL_PT_NEST, .nest = &ethtool_cable_fault_length_nest, },
+};
+
+struct ynl_policy_nest ethtool_cable_nest_nest = {
+       .max_attr = ETHTOOL_A_CABLE_NEST_MAX,
+       .table = ethtool_cable_nest_policy,
+};
+
+struct ynl_policy_attr ethtool_bitset_bits_policy[ETHTOOL_A_BITSET_BITS_MAX + 1] = {
+       [ETHTOOL_A_BITSET_BITS_BIT] = { .name = "bit", .type = YNL_PT_NEST, .nest = &ethtool_bitset_bit_nest, },
+};
+
+struct ynl_policy_nest ethtool_bitset_bits_nest = {
+       .max_attr = ETHTOOL_A_BITSET_BITS_MAX,
+       .table = ethtool_bitset_bits_policy,
+};
+
+struct ynl_policy_attr ethtool_strings_policy[ETHTOOL_A_STRINGS_MAX + 1] = {
+       [ETHTOOL_A_STRINGS_STRING] = { .name = "string", .type = YNL_PT_NEST, .nest = &ethtool_string_nest, },
+};
+
+struct ynl_policy_nest ethtool_strings_nest = {
+       .max_attr = ETHTOOL_A_STRINGS_MAX,
+       .table = ethtool_strings_policy,
+};
+
+struct ynl_policy_attr ethtool_bitset_policy[ETHTOOL_A_BITSET_MAX + 1] = {
+       [ETHTOOL_A_BITSET_NOMASK] = { .name = "nomask", .type = YNL_PT_FLAG, },
+       [ETHTOOL_A_BITSET_SIZE] = { .name = "size", .type = YNL_PT_U32, },
+       [ETHTOOL_A_BITSET_BITS] = { .name = "bits", .type = YNL_PT_NEST, .nest = &ethtool_bitset_bits_nest, },
+};
+
+struct ynl_policy_nest ethtool_bitset_nest = {
+       .max_attr = ETHTOOL_A_BITSET_MAX,
+       .table = ethtool_bitset_policy,
+};
+
+struct ynl_policy_attr ethtool_stringset_policy[ETHTOOL_A_STRINGSET_MAX + 1] = {
+       [ETHTOOL_A_STRINGSET_ID] = { .name = "id", .type = YNL_PT_U32, },
+       [ETHTOOL_A_STRINGSET_COUNT] = { .name = "count", .type = YNL_PT_U32, },
+       [ETHTOOL_A_STRINGSET_STRINGS] = { .name = "strings", .type = YNL_PT_NEST, .nest = &ethtool_strings_nest, },
+};
+
+struct ynl_policy_nest ethtool_stringset_nest = {
+       .max_attr = ETHTOOL_A_STRINGSET_MAX,
+       .table = ethtool_stringset_policy,
+};
+
+struct ynl_policy_attr ethtool_tunnel_udp_table_policy[ETHTOOL_A_TUNNEL_UDP_TABLE_MAX + 1] = {
+       [ETHTOOL_A_TUNNEL_UDP_TABLE_SIZE] = { .name = "size", .type = YNL_PT_U32, },
+       [ETHTOOL_A_TUNNEL_UDP_TABLE_TYPES] = { .name = "types", .type = YNL_PT_NEST, .nest = &ethtool_bitset_nest, },
+       [ETHTOOL_A_TUNNEL_UDP_TABLE_ENTRY] = { .name = "entry", .type = YNL_PT_NEST, .nest = &ethtool_tunnel_udp_entry_nest, },
+};
+
+struct ynl_policy_nest ethtool_tunnel_udp_table_nest = {
+       .max_attr = ETHTOOL_A_TUNNEL_UDP_TABLE_MAX,
+       .table = ethtool_tunnel_udp_table_policy,
+};
+
+struct ynl_policy_attr ethtool_stringsets_policy[ETHTOOL_A_STRINGSETS_MAX + 1] = {
+       [ETHTOOL_A_STRINGSETS_STRINGSET] = { .name = "stringset", .type = YNL_PT_NEST, .nest = &ethtool_stringset_nest, },
+};
+
+struct ynl_policy_nest ethtool_stringsets_nest = {
+       .max_attr = ETHTOOL_A_STRINGSETS_MAX,
+       .table = ethtool_stringsets_policy,
+};
+
+struct ynl_policy_attr ethtool_tunnel_udp_policy[ETHTOOL_A_TUNNEL_UDP_MAX + 1] = {
+       [ETHTOOL_A_TUNNEL_UDP_TABLE] = { .name = "table", .type = YNL_PT_NEST, .nest = &ethtool_tunnel_udp_table_nest, },
+};
+
+struct ynl_policy_nest ethtool_tunnel_udp_nest = {
+       .max_attr = ETHTOOL_A_TUNNEL_UDP_MAX,
+       .table = ethtool_tunnel_udp_policy,
+};
+
+struct ynl_policy_attr ethtool_strset_policy[ETHTOOL_A_STRSET_MAX + 1] = {
+       [ETHTOOL_A_STRSET_HEADER] = { .name = "header", .type = YNL_PT_NEST, .nest = &ethtool_header_nest, },
+       [ETHTOOL_A_STRSET_STRINGSETS] = { .name = "stringsets", .type = YNL_PT_NEST, .nest = &ethtool_stringsets_nest, },
+       [ETHTOOL_A_STRSET_COUNTS_ONLY] = { .name = "counts-only", .type = YNL_PT_FLAG, },
+};
+
+struct ynl_policy_nest ethtool_strset_nest = {
+       .max_attr = ETHTOOL_A_STRSET_MAX,
+       .table = ethtool_strset_policy,
+};
+
+struct ynl_policy_attr ethtool_linkinfo_policy[ETHTOOL_A_LINKINFO_MAX + 1] = {
+       [ETHTOOL_A_LINKINFO_HEADER] = { .name = "header", .type = YNL_PT_NEST, .nest = &ethtool_header_nest, },
+       [ETHTOOL_A_LINKINFO_PORT] = { .name = "port", .type = YNL_PT_U8, },
+       [ETHTOOL_A_LINKINFO_PHYADDR] = { .name = "phyaddr", .type = YNL_PT_U8, },
+       [ETHTOOL_A_LINKINFO_TP_MDIX] = { .name = "tp-mdix", .type = YNL_PT_U8, },
+       [ETHTOOL_A_LINKINFO_TP_MDIX_CTRL] = { .name = "tp-mdix-ctrl", .type = YNL_PT_U8, },
+       [ETHTOOL_A_LINKINFO_TRANSCEIVER] = { .name = "transceiver", .type = YNL_PT_U8, },
+};
+
+struct ynl_policy_nest ethtool_linkinfo_nest = {
+       .max_attr = ETHTOOL_A_LINKINFO_MAX,
+       .table = ethtool_linkinfo_policy,
+};
+
+struct ynl_policy_attr ethtool_linkmodes_policy[ETHTOOL_A_LINKMODES_MAX + 1] = {
+       [ETHTOOL_A_LINKMODES_HEADER] = { .name = "header", .type = YNL_PT_NEST, .nest = &ethtool_header_nest, },
+       [ETHTOOL_A_LINKMODES_AUTONEG] = { .name = "autoneg", .type = YNL_PT_U8, },
+       [ETHTOOL_A_LINKMODES_OURS] = { .name = "ours", .type = YNL_PT_NEST, .nest = &ethtool_bitset_nest, },
+       [ETHTOOL_A_LINKMODES_PEER] = { .name = "peer", .type = YNL_PT_NEST, .nest = &ethtool_bitset_nest, },
+       [ETHTOOL_A_LINKMODES_SPEED] = { .name = "speed", .type = YNL_PT_U32, },
+       [ETHTOOL_A_LINKMODES_DUPLEX] = { .name = "duplex", .type = YNL_PT_U8, },
+       [ETHTOOL_A_LINKMODES_MASTER_SLAVE_CFG] = { .name = "master-slave-cfg", .type = YNL_PT_U8, },
+       [ETHTOOL_A_LINKMODES_MASTER_SLAVE_STATE] = { .name = "master-slave-state", .type = YNL_PT_U8, },
+       [ETHTOOL_A_LINKMODES_LANES] = { .name = "lanes", .type = YNL_PT_U32, },
+       [ETHTOOL_A_LINKMODES_RATE_MATCHING] = { .name = "rate-matching", .type = YNL_PT_U8, },
+};
+
+struct ynl_policy_nest ethtool_linkmodes_nest = {
+       .max_attr = ETHTOOL_A_LINKMODES_MAX,
+       .table = ethtool_linkmodes_policy,
+};
+
+struct ynl_policy_attr ethtool_linkstate_policy[ETHTOOL_A_LINKSTATE_MAX + 1] = {
+       [ETHTOOL_A_LINKSTATE_HEADER] = { .name = "header", .type = YNL_PT_NEST, .nest = &ethtool_header_nest, },
+       [ETHTOOL_A_LINKSTATE_LINK] = { .name = "link", .type = YNL_PT_U8, },
+       [ETHTOOL_A_LINKSTATE_SQI] = { .name = "sqi", .type = YNL_PT_U32, },
+       [ETHTOOL_A_LINKSTATE_SQI_MAX] = { .name = "sqi-max", .type = YNL_PT_U32, },
+       [ETHTOOL_A_LINKSTATE_EXT_STATE] = { .name = "ext-state", .type = YNL_PT_U8, },
+       [ETHTOOL_A_LINKSTATE_EXT_SUBSTATE] = { .name = "ext-substate", .type = YNL_PT_U8, },
+       [ETHTOOL_A_LINKSTATE_EXT_DOWN_CNT] = { .name = "ext-down-cnt", .type = YNL_PT_U32, },
+};
+
+struct ynl_policy_nest ethtool_linkstate_nest = {
+       .max_attr = ETHTOOL_A_LINKSTATE_MAX,
+       .table = ethtool_linkstate_policy,
+};
+
+struct ynl_policy_attr ethtool_debug_policy[ETHTOOL_A_DEBUG_MAX + 1] = {
+       [ETHTOOL_A_DEBUG_HEADER] = { .name = "header", .type = YNL_PT_NEST, .nest = &ethtool_header_nest, },
+       [ETHTOOL_A_DEBUG_MSGMASK] = { .name = "msgmask", .type = YNL_PT_NEST, .nest = &ethtool_bitset_nest, },
+};
+
+struct ynl_policy_nest ethtool_debug_nest = {
+       .max_attr = ETHTOOL_A_DEBUG_MAX,
+       .table = ethtool_debug_policy,
+};
+
+struct ynl_policy_attr ethtool_wol_policy[ETHTOOL_A_WOL_MAX + 1] = {
+       [ETHTOOL_A_WOL_HEADER] = { .name = "header", .type = YNL_PT_NEST, .nest = &ethtool_header_nest, },
+       [ETHTOOL_A_WOL_MODES] = { .name = "modes", .type = YNL_PT_NEST, .nest = &ethtool_bitset_nest, },
+       [ETHTOOL_A_WOL_SOPASS] = { .name = "sopass", .type = YNL_PT_BINARY,},
+};
+
+struct ynl_policy_nest ethtool_wol_nest = {
+       .max_attr = ETHTOOL_A_WOL_MAX,
+       .table = ethtool_wol_policy,
+};
+
+struct ynl_policy_attr ethtool_features_policy[ETHTOOL_A_FEATURES_MAX + 1] = {
+       [ETHTOOL_A_FEATURES_HEADER] = { .name = "header", .type = YNL_PT_NEST, .nest = &ethtool_header_nest, },
+       [ETHTOOL_A_FEATURES_HW] = { .name = "hw", .type = YNL_PT_NEST, .nest = &ethtool_bitset_nest, },
+       [ETHTOOL_A_FEATURES_WANTED] = { .name = "wanted", .type = YNL_PT_NEST, .nest = &ethtool_bitset_nest, },
+       [ETHTOOL_A_FEATURES_ACTIVE] = { .name = "active", .type = YNL_PT_NEST, .nest = &ethtool_bitset_nest, },
+       [ETHTOOL_A_FEATURES_NOCHANGE] = { .name = "nochange", .type = YNL_PT_NEST, .nest = &ethtool_bitset_nest, },
+};
+
+struct ynl_policy_nest ethtool_features_nest = {
+       .max_attr = ETHTOOL_A_FEATURES_MAX,
+       .table = ethtool_features_policy,
+};
+
+struct ynl_policy_attr ethtool_privflags_policy[ETHTOOL_A_PRIVFLAGS_MAX + 1] = {
+       [ETHTOOL_A_PRIVFLAGS_HEADER] = { .name = "header", .type = YNL_PT_NEST, .nest = &ethtool_header_nest, },
+       [ETHTOOL_A_PRIVFLAGS_FLAGS] = { .name = "flags", .type = YNL_PT_NEST, .nest = &ethtool_bitset_nest, },
+};
+
+struct ynl_policy_nest ethtool_privflags_nest = {
+       .max_attr = ETHTOOL_A_PRIVFLAGS_MAX,
+       .table = ethtool_privflags_policy,
+};
+
+struct ynl_policy_attr ethtool_rings_policy[ETHTOOL_A_RINGS_MAX + 1] = {
+       [ETHTOOL_A_RINGS_HEADER] = { .name = "header", .type = YNL_PT_NEST, .nest = &ethtool_header_nest, },
+       [ETHTOOL_A_RINGS_RX_MAX] = { .name = "rx-max", .type = YNL_PT_U32, },
+       [ETHTOOL_A_RINGS_RX_MINI_MAX] = { .name = "rx-mini-max", .type = YNL_PT_U32, },
+       [ETHTOOL_A_RINGS_RX_JUMBO_MAX] = { .name = "rx-jumbo-max", .type = YNL_PT_U32, },
+       [ETHTOOL_A_RINGS_TX_MAX] = { .name = "tx-max", .type = YNL_PT_U32, },
+       [ETHTOOL_A_RINGS_RX] = { .name = "rx", .type = YNL_PT_U32, },
+       [ETHTOOL_A_RINGS_RX_MINI] = { .name = "rx-mini", .type = YNL_PT_U32, },
+       [ETHTOOL_A_RINGS_RX_JUMBO] = { .name = "rx-jumbo", .type = YNL_PT_U32, },
+       [ETHTOOL_A_RINGS_TX] = { .name = "tx", .type = YNL_PT_U32, },
+       [ETHTOOL_A_RINGS_RX_BUF_LEN] = { .name = "rx-buf-len", .type = YNL_PT_U32, },
+       [ETHTOOL_A_RINGS_TCP_DATA_SPLIT] = { .name = "tcp-data-split", .type = YNL_PT_U8, },
+       [ETHTOOL_A_RINGS_CQE_SIZE] = { .name = "cqe-size", .type = YNL_PT_U32, },
+       [ETHTOOL_A_RINGS_TX_PUSH] = { .name = "tx-push", .type = YNL_PT_U8, },
+       [ETHTOOL_A_RINGS_RX_PUSH] = { .name = "rx-push", .type = YNL_PT_U8, },
+       [ETHTOOL_A_RINGS_TX_PUSH_BUF_LEN] = { .name = "tx-push-buf-len", .type = YNL_PT_U32, },
+       [ETHTOOL_A_RINGS_TX_PUSH_BUF_LEN_MAX] = { .name = "tx-push-buf-len-max", .type = YNL_PT_U32, },
+};
+
+struct ynl_policy_nest ethtool_rings_nest = {
+       .max_attr = ETHTOOL_A_RINGS_MAX,
+       .table = ethtool_rings_policy,
+};
+
+struct ynl_policy_attr ethtool_channels_policy[ETHTOOL_A_CHANNELS_MAX + 1] = {
+       [ETHTOOL_A_CHANNELS_HEADER] = { .name = "header", .type = YNL_PT_NEST, .nest = &ethtool_header_nest, },
+       [ETHTOOL_A_CHANNELS_RX_MAX] = { .name = "rx-max", .type = YNL_PT_U32, },
+       [ETHTOOL_A_CHANNELS_TX_MAX] = { .name = "tx-max", .type = YNL_PT_U32, },
+       [ETHTOOL_A_CHANNELS_OTHER_MAX] = { .name = "other-max", .type = YNL_PT_U32, },
+       [ETHTOOL_A_CHANNELS_COMBINED_MAX] = { .name = "combined-max", .type = YNL_PT_U32, },
+       [ETHTOOL_A_CHANNELS_RX_COUNT] = { .name = "rx-count", .type = YNL_PT_U32, },
+       [ETHTOOL_A_CHANNELS_TX_COUNT] = { .name = "tx-count", .type = YNL_PT_U32, },
+       [ETHTOOL_A_CHANNELS_OTHER_COUNT] = { .name = "other-count", .type = YNL_PT_U32, },
+       [ETHTOOL_A_CHANNELS_COMBINED_COUNT] = { .name = "combined-count", .type = YNL_PT_U32, },
+};
+
+struct ynl_policy_nest ethtool_channels_nest = {
+       .max_attr = ETHTOOL_A_CHANNELS_MAX,
+       .table = ethtool_channels_policy,
+};
+
+struct ynl_policy_attr ethtool_coalesce_policy[ETHTOOL_A_COALESCE_MAX + 1] = {
+       [ETHTOOL_A_COALESCE_HEADER] = { .name = "header", .type = YNL_PT_NEST, .nest = &ethtool_header_nest, },
+       [ETHTOOL_A_COALESCE_RX_USECS] = { .name = "rx-usecs", .type = YNL_PT_U32, },
+       [ETHTOOL_A_COALESCE_RX_MAX_FRAMES] = { .name = "rx-max-frames", .type = YNL_PT_U32, },
+       [ETHTOOL_A_COALESCE_RX_USECS_IRQ] = { .name = "rx-usecs-irq", .type = YNL_PT_U32, },
+       [ETHTOOL_A_COALESCE_RX_MAX_FRAMES_IRQ] = { .name = "rx-max-frames-irq", .type = YNL_PT_U32, },
+       [ETHTOOL_A_COALESCE_TX_USECS] = { .name = "tx-usecs", .type = YNL_PT_U32, },
+       [ETHTOOL_A_COALESCE_TX_MAX_FRAMES] = { .name = "tx-max-frames", .type = YNL_PT_U32, },
+       [ETHTOOL_A_COALESCE_TX_USECS_IRQ] = { .name = "tx-usecs-irq", .type = YNL_PT_U32, },
+       [ETHTOOL_A_COALESCE_TX_MAX_FRAMES_IRQ] = { .name = "tx-max-frames-irq", .type = YNL_PT_U32, },
+       [ETHTOOL_A_COALESCE_STATS_BLOCK_USECS] = { .name = "stats-block-usecs", .type = YNL_PT_U32, },
+       [ETHTOOL_A_COALESCE_USE_ADAPTIVE_RX] = { .name = "use-adaptive-rx", .type = YNL_PT_U8, },
+       [ETHTOOL_A_COALESCE_USE_ADAPTIVE_TX] = { .name = "use-adaptive-tx", .type = YNL_PT_U8, },
+       [ETHTOOL_A_COALESCE_PKT_RATE_LOW] = { .name = "pkt-rate-low", .type = YNL_PT_U32, },
+       [ETHTOOL_A_COALESCE_RX_USECS_LOW] = { .name = "rx-usecs-low", .type = YNL_PT_U32, },
+       [ETHTOOL_A_COALESCE_RX_MAX_FRAMES_LOW] = { .name = "rx-max-frames-low", .type = YNL_PT_U32, },
+       [ETHTOOL_A_COALESCE_TX_USECS_LOW] = { .name = "tx-usecs-low", .type = YNL_PT_U32, },
+       [ETHTOOL_A_COALESCE_TX_MAX_FRAMES_LOW] = { .name = "tx-max-frames-low", .type = YNL_PT_U32, },
+       [ETHTOOL_A_COALESCE_PKT_RATE_HIGH] = { .name = "pkt-rate-high", .type = YNL_PT_U32, },
+       [ETHTOOL_A_COALESCE_RX_USECS_HIGH] = { .name = "rx-usecs-high", .type = YNL_PT_U32, },
+       [ETHTOOL_A_COALESCE_RX_MAX_FRAMES_HIGH] = { .name = "rx-max-frames-high", .type = YNL_PT_U32, },
+       [ETHTOOL_A_COALESCE_TX_USECS_HIGH] = { .name = "tx-usecs-high", .type = YNL_PT_U32, },
+       [ETHTOOL_A_COALESCE_TX_MAX_FRAMES_HIGH] = { .name = "tx-max-frames-high", .type = YNL_PT_U32, },
+       [ETHTOOL_A_COALESCE_RATE_SAMPLE_INTERVAL] = { .name = "rate-sample-interval", .type = YNL_PT_U32, },
+       [ETHTOOL_A_COALESCE_USE_CQE_MODE_TX] = { .name = "use-cqe-mode-tx", .type = YNL_PT_U8, },
+       [ETHTOOL_A_COALESCE_USE_CQE_MODE_RX] = { .name = "use-cqe-mode-rx", .type = YNL_PT_U8, },
+       [ETHTOOL_A_COALESCE_TX_AGGR_MAX_BYTES] = { .name = "tx-aggr-max-bytes", .type = YNL_PT_U32, },
+       [ETHTOOL_A_COALESCE_TX_AGGR_MAX_FRAMES] = { .name = "tx-aggr-max-frames", .type = YNL_PT_U32, },
+       [ETHTOOL_A_COALESCE_TX_AGGR_TIME_USECS] = { .name = "tx-aggr-time-usecs", .type = YNL_PT_U32, },
+};
+
+struct ynl_policy_nest ethtool_coalesce_nest = {
+       .max_attr = ETHTOOL_A_COALESCE_MAX,
+       .table = ethtool_coalesce_policy,
+};
+
+struct ynl_policy_attr ethtool_pause_policy[ETHTOOL_A_PAUSE_MAX + 1] = {
+       [ETHTOOL_A_PAUSE_HEADER] = { .name = "header", .type = YNL_PT_NEST, .nest = &ethtool_header_nest, },
+       [ETHTOOL_A_PAUSE_AUTONEG] = { .name = "autoneg", .type = YNL_PT_U8, },
+       [ETHTOOL_A_PAUSE_RX] = { .name = "rx", .type = YNL_PT_U8, },
+       [ETHTOOL_A_PAUSE_TX] = { .name = "tx", .type = YNL_PT_U8, },
+       [ETHTOOL_A_PAUSE_STATS] = { .name = "stats", .type = YNL_PT_NEST, .nest = &ethtool_pause_stat_nest, },
+       [ETHTOOL_A_PAUSE_STATS_SRC] = { .name = "stats-src", .type = YNL_PT_U32, },
+};
+
+struct ynl_policy_nest ethtool_pause_nest = {
+       .max_attr = ETHTOOL_A_PAUSE_MAX,
+       .table = ethtool_pause_policy,
+};
+
+struct ynl_policy_attr ethtool_eee_policy[ETHTOOL_A_EEE_MAX + 1] = {
+       [ETHTOOL_A_EEE_HEADER] = { .name = "header", .type = YNL_PT_NEST, .nest = &ethtool_header_nest, },
+       [ETHTOOL_A_EEE_MODES_OURS] = { .name = "modes-ours", .type = YNL_PT_NEST, .nest = &ethtool_bitset_nest, },
+       [ETHTOOL_A_EEE_MODES_PEER] = { .name = "modes-peer", .type = YNL_PT_NEST, .nest = &ethtool_bitset_nest, },
+       [ETHTOOL_A_EEE_ACTIVE] = { .name = "active", .type = YNL_PT_U8, },
+       [ETHTOOL_A_EEE_ENABLED] = { .name = "enabled", .type = YNL_PT_U8, },
+       [ETHTOOL_A_EEE_TX_LPI_ENABLED] = { .name = "tx-lpi-enabled", .type = YNL_PT_U8, },
+       [ETHTOOL_A_EEE_TX_LPI_TIMER] = { .name = "tx-lpi-timer", .type = YNL_PT_U32, },
+};
+
+struct ynl_policy_nest ethtool_eee_nest = {
+       .max_attr = ETHTOOL_A_EEE_MAX,
+       .table = ethtool_eee_policy,
+};
+
+struct ynl_policy_attr ethtool_tsinfo_policy[ETHTOOL_A_TSINFO_MAX + 1] = {
+       [ETHTOOL_A_TSINFO_HEADER] = { .name = "header", .type = YNL_PT_NEST, .nest = &ethtool_header_nest, },
+       [ETHTOOL_A_TSINFO_TIMESTAMPING] = { .name = "timestamping", .type = YNL_PT_NEST, .nest = &ethtool_bitset_nest, },
+       [ETHTOOL_A_TSINFO_TX_TYPES] = { .name = "tx-types", .type = YNL_PT_NEST, .nest = &ethtool_bitset_nest, },
+       [ETHTOOL_A_TSINFO_RX_FILTERS] = { .name = "rx-filters", .type = YNL_PT_NEST, .nest = &ethtool_bitset_nest, },
+       [ETHTOOL_A_TSINFO_PHC_INDEX] = { .name = "phc-index", .type = YNL_PT_U32, },
+};
+
+struct ynl_policy_nest ethtool_tsinfo_nest = {
+       .max_attr = ETHTOOL_A_TSINFO_MAX,
+       .table = ethtool_tsinfo_policy,
+};
+
+struct ynl_policy_attr ethtool_cable_test_policy[ETHTOOL_A_CABLE_TEST_MAX + 1] = {
+       [ETHTOOL_A_CABLE_TEST_HEADER] = { .name = "header", .type = YNL_PT_NEST, .nest = &ethtool_header_nest, },
+};
+
+struct ynl_policy_nest ethtool_cable_test_nest = {
+       .max_attr = ETHTOOL_A_CABLE_TEST_MAX,
+       .table = ethtool_cable_test_policy,
+};
+
+struct ynl_policy_attr ethtool_cable_test_ntf_policy[ETHTOOL_A_CABLE_TEST_NTF_MAX + 1] = {
+       [ETHTOOL_A_CABLE_TEST_NTF_HEADER] = { .name = "header", .type = YNL_PT_NEST, .nest = &ethtool_header_nest, },
+       [ETHTOOL_A_CABLE_TEST_NTF_STATUS] = { .name = "status", .type = YNL_PT_U8, },
+       [ETHTOOL_A_CABLE_TEST_NTF_NEST] = { .name = "nest", .type = YNL_PT_NEST, .nest = &ethtool_cable_nest_nest, },
+};
+
+struct ynl_policy_nest ethtool_cable_test_ntf_nest = {
+       .max_attr = ETHTOOL_A_CABLE_TEST_NTF_MAX,
+       .table = ethtool_cable_test_ntf_policy,
+};
+
+struct ynl_policy_attr ethtool_cable_test_tdr_policy[ETHTOOL_A_CABLE_TEST_TDR_MAX + 1] = {
+       [ETHTOOL_A_CABLE_TEST_TDR_HEADER] = { .name = "header", .type = YNL_PT_NEST, .nest = &ethtool_header_nest, },
+       [ETHTOOL_A_CABLE_TEST_TDR_CFG] = { .name = "cfg", .type = YNL_PT_NEST, .nest = &ethtool_cable_test_tdr_cfg_nest, },
+};
+
+struct ynl_policy_nest ethtool_cable_test_tdr_nest = {
+       .max_attr = ETHTOOL_A_CABLE_TEST_TDR_MAX,
+       .table = ethtool_cable_test_tdr_policy,
+};
+
+struct ynl_policy_attr ethtool_cable_test_tdr_ntf_policy[ETHTOOL_A_CABLE_TEST_TDR_NTF_MAX + 1] = {
+       [ETHTOOL_A_CABLE_TEST_TDR_NTF_HEADER] = { .name = "header", .type = YNL_PT_NEST, .nest = &ethtool_header_nest, },
+       [ETHTOOL_A_CABLE_TEST_TDR_NTF_STATUS] = { .name = "status", .type = YNL_PT_U8, },
+       [ETHTOOL_A_CABLE_TEST_TDR_NTF_NEST] = { .name = "nest", .type = YNL_PT_NEST, .nest = &ethtool_cable_nest_nest, },
+};
+
+struct ynl_policy_nest ethtool_cable_test_tdr_ntf_nest = {
+       .max_attr = ETHTOOL_A_CABLE_TEST_TDR_NTF_MAX,
+       .table = ethtool_cable_test_tdr_ntf_policy,
+};
+
+struct ynl_policy_attr ethtool_tunnel_info_policy[ETHTOOL_A_TUNNEL_INFO_MAX + 1] = {
+       [ETHTOOL_A_TUNNEL_INFO_HEADER] = { .name = "header", .type = YNL_PT_NEST, .nest = &ethtool_header_nest, },
+       [ETHTOOL_A_TUNNEL_INFO_UDP_PORTS] = { .name = "udp-ports", .type = YNL_PT_NEST, .nest = &ethtool_tunnel_udp_nest, },
+};
+
+struct ynl_policy_nest ethtool_tunnel_info_nest = {
+       .max_attr = ETHTOOL_A_TUNNEL_INFO_MAX,
+       .table = ethtool_tunnel_info_policy,
+};
+
+struct ynl_policy_attr ethtool_fec_policy[ETHTOOL_A_FEC_MAX + 1] = {
+       [ETHTOOL_A_FEC_HEADER] = { .name = "header", .type = YNL_PT_NEST, .nest = &ethtool_header_nest, },
+       [ETHTOOL_A_FEC_MODES] = { .name = "modes", .type = YNL_PT_NEST, .nest = &ethtool_bitset_nest, },
+       [ETHTOOL_A_FEC_AUTO] = { .name = "auto", .type = YNL_PT_U8, },
+       [ETHTOOL_A_FEC_ACTIVE] = { .name = "active", .type = YNL_PT_U32, },
+       [ETHTOOL_A_FEC_STATS] = { .name = "stats", .type = YNL_PT_NEST, .nest = &ethtool_fec_stat_nest, },
+};
+
+struct ynl_policy_nest ethtool_fec_nest = {
+       .max_attr = ETHTOOL_A_FEC_MAX,
+       .table = ethtool_fec_policy,
+};
+
+struct ynl_policy_attr ethtool_module_eeprom_policy[ETHTOOL_A_MODULE_EEPROM_MAX + 1] = {
+       [ETHTOOL_A_MODULE_EEPROM_HEADER] = { .name = "header", .type = YNL_PT_NEST, .nest = &ethtool_header_nest, },
+       [ETHTOOL_A_MODULE_EEPROM_OFFSET] = { .name = "offset", .type = YNL_PT_U32, },
+       [ETHTOOL_A_MODULE_EEPROM_LENGTH] = { .name = "length", .type = YNL_PT_U32, },
+       [ETHTOOL_A_MODULE_EEPROM_PAGE] = { .name = "page", .type = YNL_PT_U8, },
+       [ETHTOOL_A_MODULE_EEPROM_BANK] = { .name = "bank", .type = YNL_PT_U8, },
+       [ETHTOOL_A_MODULE_EEPROM_I2C_ADDRESS] = { .name = "i2c-address", .type = YNL_PT_U8, },
+       [ETHTOOL_A_MODULE_EEPROM_DATA] = { .name = "data", .type = YNL_PT_BINARY,},
+};
+
+struct ynl_policy_nest ethtool_module_eeprom_nest = {
+       .max_attr = ETHTOOL_A_MODULE_EEPROM_MAX,
+       .table = ethtool_module_eeprom_policy,
+};
+
+struct ynl_policy_attr ethtool_phc_vclocks_policy[ETHTOOL_A_PHC_VCLOCKS_MAX + 1] = {
+       [ETHTOOL_A_PHC_VCLOCKS_HEADER] = { .name = "header", .type = YNL_PT_NEST, .nest = &ethtool_header_nest, },
+       [ETHTOOL_A_PHC_VCLOCKS_NUM] = { .name = "num", .type = YNL_PT_U32, },
+       [ETHTOOL_A_PHC_VCLOCKS_INDEX] = { .name = "index", .type = YNL_PT_BINARY,},
+};
+
+struct ynl_policy_nest ethtool_phc_vclocks_nest = {
+       .max_attr = ETHTOOL_A_PHC_VCLOCKS_MAX,
+       .table = ethtool_phc_vclocks_policy,
+};
+
+struct ynl_policy_attr ethtool_module_policy[ETHTOOL_A_MODULE_MAX + 1] = {
+       [ETHTOOL_A_MODULE_HEADER] = { .name = "header", .type = YNL_PT_NEST, .nest = &ethtool_header_nest, },
+       [ETHTOOL_A_MODULE_POWER_MODE_POLICY] = { .name = "power-mode-policy", .type = YNL_PT_U8, },
+       [ETHTOOL_A_MODULE_POWER_MODE] = { .name = "power-mode", .type = YNL_PT_U8, },
+};
+
+struct ynl_policy_nest ethtool_module_nest = {
+       .max_attr = ETHTOOL_A_MODULE_MAX,
+       .table = ethtool_module_policy,
+};
+
+struct ynl_policy_attr ethtool_pse_policy[ETHTOOL_A_PSE_MAX + 1] = {
+       [ETHTOOL_A_PSE_HEADER] = { .name = "header", .type = YNL_PT_NEST, .nest = &ethtool_header_nest, },
+       [ETHTOOL_A_PODL_PSE_ADMIN_STATE] = { .name = "admin-state", .type = YNL_PT_U32, },
+       [ETHTOOL_A_PODL_PSE_ADMIN_CONTROL] = { .name = "admin-control", .type = YNL_PT_U32, },
+       [ETHTOOL_A_PODL_PSE_PW_D_STATUS] = { .name = "pw-d-status", .type = YNL_PT_U32, },
+};
+
+struct ynl_policy_nest ethtool_pse_nest = {
+       .max_attr = ETHTOOL_A_PSE_MAX,
+       .table = ethtool_pse_policy,
+};
+
+struct ynl_policy_attr ethtool_rss_policy[ETHTOOL_A_RSS_MAX + 1] = {
+       [ETHTOOL_A_RSS_HEADER] = { .name = "header", .type = YNL_PT_NEST, .nest = &ethtool_header_nest, },
+       [ETHTOOL_A_RSS_CONTEXT] = { .name = "context", .type = YNL_PT_U32, },
+       [ETHTOOL_A_RSS_HFUNC] = { .name = "hfunc", .type = YNL_PT_U32, },
+       [ETHTOOL_A_RSS_INDIR] = { .name = "indir", .type = YNL_PT_BINARY,},
+       [ETHTOOL_A_RSS_HKEY] = { .name = "hkey", .type = YNL_PT_BINARY,},
+};
+
+struct ynl_policy_nest ethtool_rss_nest = {
+       .max_attr = ETHTOOL_A_RSS_MAX,
+       .table = ethtool_rss_policy,
+};
+
+struct ynl_policy_attr ethtool_plca_policy[ETHTOOL_A_PLCA_MAX + 1] = {
+       [ETHTOOL_A_PLCA_HEADER] = { .name = "header", .type = YNL_PT_NEST, .nest = &ethtool_header_nest, },
+       [ETHTOOL_A_PLCA_VERSION] = { .name = "version", .type = YNL_PT_U16, },
+       [ETHTOOL_A_PLCA_ENABLED] = { .name = "enabled", .type = YNL_PT_U8, },
+       [ETHTOOL_A_PLCA_STATUS] = { .name = "status", .type = YNL_PT_U8, },
+       [ETHTOOL_A_PLCA_NODE_CNT] = { .name = "node-cnt", .type = YNL_PT_U32, },
+       [ETHTOOL_A_PLCA_NODE_ID] = { .name = "node-id", .type = YNL_PT_U32, },
+       [ETHTOOL_A_PLCA_TO_TMR] = { .name = "to-tmr", .type = YNL_PT_U32, },
+       [ETHTOOL_A_PLCA_BURST_CNT] = { .name = "burst-cnt", .type = YNL_PT_U32, },
+       [ETHTOOL_A_PLCA_BURST_TMR] = { .name = "burst-tmr", .type = YNL_PT_U32, },
+};
+
+struct ynl_policy_nest ethtool_plca_nest = {
+       .max_attr = ETHTOOL_A_PLCA_MAX,
+       .table = ethtool_plca_policy,
+};
+
+struct ynl_policy_attr ethtool_mm_policy[ETHTOOL_A_MM_MAX + 1] = {
+       [ETHTOOL_A_MM_HEADER] = { .name = "header", .type = YNL_PT_NEST, .nest = &ethtool_header_nest, },
+       [ETHTOOL_A_MM_PMAC_ENABLED] = { .name = "pmac-enabled", .type = YNL_PT_U8, },
+       [ETHTOOL_A_MM_TX_ENABLED] = { .name = "tx-enabled", .type = YNL_PT_U8, },
+       [ETHTOOL_A_MM_TX_ACTIVE] = { .name = "tx-active", .type = YNL_PT_U8, },
+       [ETHTOOL_A_MM_TX_MIN_FRAG_SIZE] = { .name = "tx-min-frag-size", .type = YNL_PT_U32, },
+       [ETHTOOL_A_MM_RX_MIN_FRAG_SIZE] = { .name = "rx-min-frag-size", .type = YNL_PT_U32, },
+       [ETHTOOL_A_MM_VERIFY_ENABLED] = { .name = "verify-enabled", .type = YNL_PT_U8, },
+       [ETHTOOL_A_MM_VERIFY_STATUS] = { .name = "verify-status", .type = YNL_PT_U8, },
+       [ETHTOOL_A_MM_VERIFY_TIME] = { .name = "verify-time", .type = YNL_PT_U32, },
+       [ETHTOOL_A_MM_MAX_VERIFY_TIME] = { .name = "max-verify-time", .type = YNL_PT_U32, },
+       [ETHTOOL_A_MM_STATS] = { .name = "stats", .type = YNL_PT_NEST, .nest = &ethtool_mm_stat_nest, },
+};
+
+struct ynl_policy_nest ethtool_mm_nest = {
+       .max_attr = ETHTOOL_A_MM_MAX,
+       .table = ethtool_mm_policy,
+};
+
+/* Common nested types */
+void ethtool_header_free(struct ethtool_header *obj)
+{
+       free(obj->dev_name);
+}
+
+int ethtool_header_put(struct nlmsghdr *nlh, unsigned int attr_type,
+                      struct ethtool_header *obj)
+{
+       struct nlattr *nest;
+
+       nest = mnl_attr_nest_start(nlh, attr_type);
+       if (obj->_present.dev_index)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_HEADER_DEV_INDEX, obj->dev_index);
+       if (obj->_present.dev_name_len)
+               mnl_attr_put_strz(nlh, ETHTOOL_A_HEADER_DEV_NAME, obj->dev_name);
+       if (obj->_present.flags)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_HEADER_FLAGS, obj->flags);
+       mnl_attr_nest_end(nlh, nest);
+
+       return 0;
+}
+
+int ethtool_header_parse(struct ynl_parse_arg *yarg,
+                        const struct nlattr *nested)
+{
+       struct ethtool_header *dst = yarg->data;
+       const struct nlattr *attr;
+
+       mnl_attr_for_each_nested(attr, nested) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_HEADER_DEV_INDEX) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.dev_index = 1;
+                       dst->dev_index = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_HEADER_DEV_NAME) {
+                       unsigned int len;
+
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+
+                       len = strnlen(mnl_attr_get_str(attr), mnl_attr_get_payload_len(attr));
+                       dst->_present.dev_name_len = len;
+                       dst->dev_name = malloc(len + 1);
+                       memcpy(dst->dev_name, mnl_attr_get_str(attr), len);
+                       dst->dev_name[len] = 0;
+               } else if (type == ETHTOOL_A_HEADER_FLAGS) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.flags = 1;
+                       dst->flags = mnl_attr_get_u32(attr);
+               }
+       }
+
+       return 0;
+}
+
+void ethtool_pause_stat_free(struct ethtool_pause_stat *obj)
+{
+}
+
+int ethtool_pause_stat_put(struct nlmsghdr *nlh, unsigned int attr_type,
+                          struct ethtool_pause_stat *obj)
+{
+       struct nlattr *nest;
+
+       nest = mnl_attr_nest_start(nlh, attr_type);
+       if (obj->_present.tx_frames)
+               mnl_attr_put_u64(nlh, ETHTOOL_A_PAUSE_STAT_TX_FRAMES, obj->tx_frames);
+       if (obj->_present.rx_frames)
+               mnl_attr_put_u64(nlh, ETHTOOL_A_PAUSE_STAT_RX_FRAMES, obj->rx_frames);
+       mnl_attr_nest_end(nlh, nest);
+
+       return 0;
+}
+
+int ethtool_pause_stat_parse(struct ynl_parse_arg *yarg,
+                            const struct nlattr *nested)
+{
+       struct ethtool_pause_stat *dst = yarg->data;
+       const struct nlattr *attr;
+
+       mnl_attr_for_each_nested(attr, nested) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_PAUSE_STAT_TX_FRAMES) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.tx_frames = 1;
+                       dst->tx_frames = mnl_attr_get_u64(attr);
+               } else if (type == ETHTOOL_A_PAUSE_STAT_RX_FRAMES) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.rx_frames = 1;
+                       dst->rx_frames = mnl_attr_get_u64(attr);
+               }
+       }
+
+       return 0;
+}
+
+void ethtool_cable_test_tdr_cfg_free(struct ethtool_cable_test_tdr_cfg *obj)
+{
+}
+
+void ethtool_fec_stat_free(struct ethtool_fec_stat *obj)
+{
+       free(obj->corrected);
+       free(obj->uncorr);
+       free(obj->corr_bits);
+}
+
+int ethtool_fec_stat_put(struct nlmsghdr *nlh, unsigned int attr_type,
+                        struct ethtool_fec_stat *obj)
+{
+       struct nlattr *nest;
+
+       nest = mnl_attr_nest_start(nlh, attr_type);
+       if (obj->_present.corrected_len)
+               mnl_attr_put(nlh, ETHTOOL_A_FEC_STAT_CORRECTED, obj->_present.corrected_len, obj->corrected);
+       if (obj->_present.uncorr_len)
+               mnl_attr_put(nlh, ETHTOOL_A_FEC_STAT_UNCORR, obj->_present.uncorr_len, obj->uncorr);
+       if (obj->_present.corr_bits_len)
+               mnl_attr_put(nlh, ETHTOOL_A_FEC_STAT_CORR_BITS, obj->_present.corr_bits_len, obj->corr_bits);
+       mnl_attr_nest_end(nlh, nest);
+
+       return 0;
+}
+
+int ethtool_fec_stat_parse(struct ynl_parse_arg *yarg,
+                          const struct nlattr *nested)
+{
+       struct ethtool_fec_stat *dst = yarg->data;
+       const struct nlattr *attr;
+
+       mnl_attr_for_each_nested(attr, nested) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_FEC_STAT_CORRECTED) {
+                       unsigned int len;
+
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+
+                       len = mnl_attr_get_payload_len(attr);
+                       dst->_present.corrected_len = len;
+                       dst->corrected = malloc(len);
+                       memcpy(dst->corrected, mnl_attr_get_payload(attr), len);
+               } else if (type == ETHTOOL_A_FEC_STAT_UNCORR) {
+                       unsigned int len;
+
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+
+                       len = mnl_attr_get_payload_len(attr);
+                       dst->_present.uncorr_len = len;
+                       dst->uncorr = malloc(len);
+                       memcpy(dst->uncorr, mnl_attr_get_payload(attr), len);
+               } else if (type == ETHTOOL_A_FEC_STAT_CORR_BITS) {
+                       unsigned int len;
+
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+
+                       len = mnl_attr_get_payload_len(attr);
+                       dst->_present.corr_bits_len = len;
+                       dst->corr_bits = malloc(len);
+                       memcpy(dst->corr_bits, mnl_attr_get_payload(attr), len);
+               }
+       }
+
+       return 0;
+}
+
+void ethtool_mm_stat_free(struct ethtool_mm_stat *obj)
+{
+}
+
+int ethtool_mm_stat_parse(struct ynl_parse_arg *yarg,
+                         const struct nlattr *nested)
+{
+       struct ethtool_mm_stat *dst = yarg->data;
+       const struct nlattr *attr;
+
+       mnl_attr_for_each_nested(attr, nested) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_MM_STAT_REASSEMBLY_ERRORS) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.reassembly_errors = 1;
+                       dst->reassembly_errors = mnl_attr_get_u64(attr);
+               } else if (type == ETHTOOL_A_MM_STAT_SMD_ERRORS) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.smd_errors = 1;
+                       dst->smd_errors = mnl_attr_get_u64(attr);
+               } else if (type == ETHTOOL_A_MM_STAT_REASSEMBLY_OK) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.reassembly_ok = 1;
+                       dst->reassembly_ok = mnl_attr_get_u64(attr);
+               } else if (type == ETHTOOL_A_MM_STAT_RX_FRAG_COUNT) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.rx_frag_count = 1;
+                       dst->rx_frag_count = mnl_attr_get_u64(attr);
+               } else if (type == ETHTOOL_A_MM_STAT_TX_FRAG_COUNT) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.tx_frag_count = 1;
+                       dst->tx_frag_count = mnl_attr_get_u64(attr);
+               } else if (type == ETHTOOL_A_MM_STAT_HOLD_COUNT) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.hold_count = 1;
+                       dst->hold_count = mnl_attr_get_u64(attr);
+               }
+       }
+
+       return 0;
+}
+
+void ethtool_cable_result_free(struct ethtool_cable_result *obj)
+{
+}
+
+int ethtool_cable_result_parse(struct ynl_parse_arg *yarg,
+                              const struct nlattr *nested)
+{
+       struct ethtool_cable_result *dst = yarg->data;
+       const struct nlattr *attr;
+
+       mnl_attr_for_each_nested(attr, nested) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_CABLE_RESULT_PAIR) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.pair = 1;
+                       dst->pair = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_CABLE_RESULT_CODE) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.code = 1;
+                       dst->code = mnl_attr_get_u8(attr);
+               }
+       }
+
+       return 0;
+}
+
+void ethtool_cable_fault_length_free(struct ethtool_cable_fault_length *obj)
+{
+}
+
+int ethtool_cable_fault_length_parse(struct ynl_parse_arg *yarg,
+                                    const struct nlattr *nested)
+{
+       struct ethtool_cable_fault_length *dst = yarg->data;
+       const struct nlattr *attr;
+
+       mnl_attr_for_each_nested(attr, nested) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_CABLE_FAULT_LENGTH_PAIR) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.pair = 1;
+                       dst->pair = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_CABLE_FAULT_LENGTH_CM) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.cm = 1;
+                       dst->cm = mnl_attr_get_u32(attr);
+               }
+       }
+
+       return 0;
+}
+
+void ethtool_bitset_bit_free(struct ethtool_bitset_bit *obj)
+{
+       free(obj->name);
+}
+
+int ethtool_bitset_bit_put(struct nlmsghdr *nlh, unsigned int attr_type,
+                          struct ethtool_bitset_bit *obj)
+{
+       struct nlattr *nest;
+
+       nest = mnl_attr_nest_start(nlh, attr_type);
+       if (obj->_present.index)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_BITSET_BIT_INDEX, obj->index);
+       if (obj->_present.name_len)
+               mnl_attr_put_strz(nlh, ETHTOOL_A_BITSET_BIT_NAME, obj->name);
+       if (obj->_present.value)
+               mnl_attr_put(nlh, ETHTOOL_A_BITSET_BIT_VALUE, 0, NULL);
+       mnl_attr_nest_end(nlh, nest);
+
+       return 0;
+}
+
+int ethtool_bitset_bit_parse(struct ynl_parse_arg *yarg,
+                            const struct nlattr *nested)
+{
+       struct ethtool_bitset_bit *dst = yarg->data;
+       const struct nlattr *attr;
+
+       mnl_attr_for_each_nested(attr, nested) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_BITSET_BIT_INDEX) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.index = 1;
+                       dst->index = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_BITSET_BIT_NAME) {
+                       unsigned int len;
+
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+
+                       len = strnlen(mnl_attr_get_str(attr), mnl_attr_get_payload_len(attr));
+                       dst->_present.name_len = len;
+                       dst->name = malloc(len + 1);
+                       memcpy(dst->name, mnl_attr_get_str(attr), len);
+                       dst->name[len] = 0;
+               } else if (type == ETHTOOL_A_BITSET_BIT_VALUE) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.value = 1;
+               }
+       }
+
+       return 0;
+}
+
+void ethtool_tunnel_udp_entry_free(struct ethtool_tunnel_udp_entry *obj)
+{
+}
+
+int ethtool_tunnel_udp_entry_parse(struct ynl_parse_arg *yarg,
+                                  const struct nlattr *nested)
+{
+       struct ethtool_tunnel_udp_entry *dst = yarg->data;
+       const struct nlattr *attr;
+
+       mnl_attr_for_each_nested(attr, nested) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_TUNNEL_UDP_ENTRY_PORT) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.port = 1;
+                       dst->port = mnl_attr_get_u16(attr);
+               } else if (type == ETHTOOL_A_TUNNEL_UDP_ENTRY_TYPE) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.type = 1;
+                       dst->type = mnl_attr_get_u32(attr);
+               }
+       }
+
+       return 0;
+}
+
+void ethtool_string_free(struct ethtool_string *obj)
+{
+       free(obj->value);
+}
+
+int ethtool_string_put(struct nlmsghdr *nlh, unsigned int attr_type,
+                      struct ethtool_string *obj)
+{
+       struct nlattr *nest;
+
+       nest = mnl_attr_nest_start(nlh, attr_type);
+       if (obj->_present.index)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_STRING_INDEX, obj->index);
+       if (obj->_present.value_len)
+               mnl_attr_put_strz(nlh, ETHTOOL_A_STRING_VALUE, obj->value);
+       mnl_attr_nest_end(nlh, nest);
+
+       return 0;
+}
+
+int ethtool_string_parse(struct ynl_parse_arg *yarg,
+                        const struct nlattr *nested)
+{
+       struct ethtool_string *dst = yarg->data;
+       const struct nlattr *attr;
+
+       mnl_attr_for_each_nested(attr, nested) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_STRING_INDEX) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.index = 1;
+                       dst->index = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_STRING_VALUE) {
+                       unsigned int len;
+
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+
+                       len = strnlen(mnl_attr_get_str(attr), mnl_attr_get_payload_len(attr));
+                       dst->_present.value_len = len;
+                       dst->value = malloc(len + 1);
+                       memcpy(dst->value, mnl_attr_get_str(attr), len);
+                       dst->value[len] = 0;
+               }
+       }
+
+       return 0;
+}
+
+void ethtool_cable_nest_free(struct ethtool_cable_nest *obj)
+{
+       ethtool_cable_result_free(&obj->result);
+       ethtool_cable_fault_length_free(&obj->fault_length);
+}
+
+int ethtool_cable_nest_parse(struct ynl_parse_arg *yarg,
+                            const struct nlattr *nested)
+{
+       struct ethtool_cable_nest *dst = yarg->data;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+
+       parg.ys = yarg->ys;
+
+       mnl_attr_for_each_nested(attr, nested) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_CABLE_NEST_RESULT) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.result = 1;
+
+                       parg.rsp_policy = &ethtool_cable_result_nest;
+                       parg.data = &dst->result;
+                       if (ethtool_cable_result_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_CABLE_NEST_FAULT_LENGTH) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.fault_length = 1;
+
+                       parg.rsp_policy = &ethtool_cable_fault_length_nest;
+                       parg.data = &dst->fault_length;
+                       if (ethtool_cable_fault_length_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               }
+       }
+
+       return 0;
+}
+
+void ethtool_bitset_bits_free(struct ethtool_bitset_bits *obj)
+{
+       unsigned int i;
+
+       for (i = 0; i < obj->n_bit; i++)
+               ethtool_bitset_bit_free(&obj->bit[i]);
+       free(obj->bit);
+}
+
+int ethtool_bitset_bits_put(struct nlmsghdr *nlh, unsigned int attr_type,
+                           struct ethtool_bitset_bits *obj)
+{
+       struct nlattr *nest;
+
+       nest = mnl_attr_nest_start(nlh, attr_type);
+       for (unsigned int i = 0; i < obj->n_bit; i++)
+               ethtool_bitset_bit_put(nlh, ETHTOOL_A_BITSET_BITS_BIT, &obj->bit[i]);
+       mnl_attr_nest_end(nlh, nest);
+
+       return 0;
+}
+
+int ethtool_bitset_bits_parse(struct ynl_parse_arg *yarg,
+                             const struct nlattr *nested)
+{
+       struct ethtool_bitset_bits *dst = yarg->data;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+       unsigned int n_bit = 0;
+       int i;
+
+       parg.ys = yarg->ys;
+
+       if (dst->bit)
+               return ynl_error_parse(yarg, "attribute already present (bitset-bits.bit)");
+
+       mnl_attr_for_each_nested(attr, nested) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_BITSET_BITS_BIT) {
+                       n_bit++;
+               }
+       }
+
+       if (n_bit) {
+               dst->bit = calloc(n_bit, sizeof(*dst->bit));
+               dst->n_bit = n_bit;
+               i = 0;
+               parg.rsp_policy = &ethtool_bitset_bit_nest;
+               mnl_attr_for_each_nested(attr, nested) {
+                       if (mnl_attr_get_type(attr) == ETHTOOL_A_BITSET_BITS_BIT) {
+                               parg.data = &dst->bit[i];
+                               if (ethtool_bitset_bit_parse(&parg, attr))
+                                       return MNL_CB_ERROR;
+                               i++;
+                       }
+               }
+       }
+
+       return 0;
+}
+
+void ethtool_strings_free(struct ethtool_strings *obj)
+{
+       unsigned int i;
+
+       for (i = 0; i < obj->n_string; i++)
+               ethtool_string_free(&obj->string[i]);
+       free(obj->string);
+}
+
+int ethtool_strings_put(struct nlmsghdr *nlh, unsigned int attr_type,
+                       struct ethtool_strings *obj)
+{
+       struct nlattr *nest;
+
+       nest = mnl_attr_nest_start(nlh, attr_type);
+       for (unsigned int i = 0; i < obj->n_string; i++)
+               ethtool_string_put(nlh, ETHTOOL_A_STRINGS_STRING, &obj->string[i]);
+       mnl_attr_nest_end(nlh, nest);
+
+       return 0;
+}
+
+int ethtool_strings_parse(struct ynl_parse_arg *yarg,
+                         const struct nlattr *nested)
+{
+       struct ethtool_strings *dst = yarg->data;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+       unsigned int n_string = 0;
+       int i;
+
+       parg.ys = yarg->ys;
+
+       if (dst->string)
+               return ynl_error_parse(yarg, "attribute already present (strings.string)");
+
+       mnl_attr_for_each_nested(attr, nested) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_STRINGS_STRING) {
+                       n_string++;
+               }
+       }
+
+       if (n_string) {
+               dst->string = calloc(n_string, sizeof(*dst->string));
+               dst->n_string = n_string;
+               i = 0;
+               parg.rsp_policy = &ethtool_string_nest;
+               mnl_attr_for_each_nested(attr, nested) {
+                       if (mnl_attr_get_type(attr) == ETHTOOL_A_STRINGS_STRING) {
+                               parg.data = &dst->string[i];
+                               if (ethtool_string_parse(&parg, attr))
+                                       return MNL_CB_ERROR;
+                               i++;
+                       }
+               }
+       }
+
+       return 0;
+}
+
+void ethtool_bitset_free(struct ethtool_bitset *obj)
+{
+       ethtool_bitset_bits_free(&obj->bits);
+}
+
+int ethtool_bitset_put(struct nlmsghdr *nlh, unsigned int attr_type,
+                      struct ethtool_bitset *obj)
+{
+       struct nlattr *nest;
+
+       nest = mnl_attr_nest_start(nlh, attr_type);
+       if (obj->_present.nomask)
+               mnl_attr_put(nlh, ETHTOOL_A_BITSET_NOMASK, 0, NULL);
+       if (obj->_present.size)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_BITSET_SIZE, obj->size);
+       if (obj->_present.bits)
+               ethtool_bitset_bits_put(nlh, ETHTOOL_A_BITSET_BITS, &obj->bits);
+       mnl_attr_nest_end(nlh, nest);
+
+       return 0;
+}
+
+int ethtool_bitset_parse(struct ynl_parse_arg *yarg,
+                        const struct nlattr *nested)
+{
+       struct ethtool_bitset *dst = yarg->data;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+
+       parg.ys = yarg->ys;
+
+       mnl_attr_for_each_nested(attr, nested) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_BITSET_NOMASK) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.nomask = 1;
+               } else if (type == ETHTOOL_A_BITSET_SIZE) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.size = 1;
+                       dst->size = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_BITSET_BITS) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.bits = 1;
+
+                       parg.rsp_policy = &ethtool_bitset_bits_nest;
+                       parg.data = &dst->bits;
+                       if (ethtool_bitset_bits_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               }
+       }
+
+       return 0;
+}
+
+void ethtool_stringset_free(struct ethtool_stringset_ *obj)
+{
+       unsigned int i;
+
+       for (i = 0; i < obj->n_strings; i++)
+               ethtool_strings_free(&obj->strings[i]);
+       free(obj->strings);
+}
+
+int ethtool_stringset_put(struct nlmsghdr *nlh, unsigned int attr_type,
+                         struct ethtool_stringset_ *obj)
+{
+       struct nlattr *nest;
+
+       nest = mnl_attr_nest_start(nlh, attr_type);
+       if (obj->_present.id)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_STRINGSET_ID, obj->id);
+       if (obj->_present.count)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_STRINGSET_COUNT, obj->count);
+       for (unsigned int i = 0; i < obj->n_strings; i++)
+               ethtool_strings_put(nlh, ETHTOOL_A_STRINGSET_STRINGS, &obj->strings[i]);
+       mnl_attr_nest_end(nlh, nest);
+
+       return 0;
+}
+
+int ethtool_stringset_parse(struct ynl_parse_arg *yarg,
+                           const struct nlattr *nested)
+{
+       struct ethtool_stringset_ *dst = yarg->data;
+       unsigned int n_strings = 0;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+       int i;
+
+       parg.ys = yarg->ys;
+
+       if (dst->strings)
+               return ynl_error_parse(yarg, "attribute already present (stringset.strings)");
+
+       mnl_attr_for_each_nested(attr, nested) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_STRINGSET_ID) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.id = 1;
+                       dst->id = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_STRINGSET_COUNT) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.count = 1;
+                       dst->count = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_STRINGSET_STRINGS) {
+                       n_strings++;
+               }
+       }
+
+       if (n_strings) {
+               dst->strings = calloc(n_strings, sizeof(*dst->strings));
+               dst->n_strings = n_strings;
+               i = 0;
+               parg.rsp_policy = &ethtool_strings_nest;
+               mnl_attr_for_each_nested(attr, nested) {
+                       if (mnl_attr_get_type(attr) == ETHTOOL_A_STRINGSET_STRINGS) {
+                               parg.data = &dst->strings[i];
+                               if (ethtool_strings_parse(&parg, attr))
+                                       return MNL_CB_ERROR;
+                               i++;
+                       }
+               }
+       }
+
+       return 0;
+}
+
+void ethtool_tunnel_udp_table_free(struct ethtool_tunnel_udp_table *obj)
+{
+       unsigned int i;
+
+       ethtool_bitset_free(&obj->types);
+       for (i = 0; i < obj->n_entry; i++)
+               ethtool_tunnel_udp_entry_free(&obj->entry[i]);
+       free(obj->entry);
+}
+
+int ethtool_tunnel_udp_table_parse(struct ynl_parse_arg *yarg,
+                                  const struct nlattr *nested)
+{
+       struct ethtool_tunnel_udp_table *dst = yarg->data;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+       unsigned int n_entry = 0;
+       int i;
+
+       parg.ys = yarg->ys;
+
+       if (dst->entry)
+               return ynl_error_parse(yarg, "attribute already present (tunnel-udp-table.entry)");
+
+       mnl_attr_for_each_nested(attr, nested) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_TUNNEL_UDP_TABLE_SIZE) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.size = 1;
+                       dst->size = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_TUNNEL_UDP_TABLE_TYPES) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.types = 1;
+
+                       parg.rsp_policy = &ethtool_bitset_nest;
+                       parg.data = &dst->types;
+                       if (ethtool_bitset_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_TUNNEL_UDP_TABLE_ENTRY) {
+                       n_entry++;
+               }
+       }
+
+       if (n_entry) {
+               dst->entry = calloc(n_entry, sizeof(*dst->entry));
+               dst->n_entry = n_entry;
+               i = 0;
+               parg.rsp_policy = &ethtool_tunnel_udp_entry_nest;
+               mnl_attr_for_each_nested(attr, nested) {
+                       if (mnl_attr_get_type(attr) == ETHTOOL_A_TUNNEL_UDP_TABLE_ENTRY) {
+                               parg.data = &dst->entry[i];
+                               if (ethtool_tunnel_udp_entry_parse(&parg, attr))
+                                       return MNL_CB_ERROR;
+                               i++;
+                       }
+               }
+       }
+
+       return 0;
+}
+
+void ethtool_stringsets_free(struct ethtool_stringsets *obj)
+{
+       unsigned int i;
+
+       for (i = 0; i < obj->n_stringset; i++)
+               ethtool_stringset_free(&obj->stringset[i]);
+       free(obj->stringset);
+}
+
+int ethtool_stringsets_put(struct nlmsghdr *nlh, unsigned int attr_type,
+                          struct ethtool_stringsets *obj)
+{
+       struct nlattr *nest;
+
+       nest = mnl_attr_nest_start(nlh, attr_type);
+       for (unsigned int i = 0; i < obj->n_stringset; i++)
+               ethtool_stringset_put(nlh, ETHTOOL_A_STRINGSETS_STRINGSET, &obj->stringset[i]);
+       mnl_attr_nest_end(nlh, nest);
+
+       return 0;
+}
+
+int ethtool_stringsets_parse(struct ynl_parse_arg *yarg,
+                            const struct nlattr *nested)
+{
+       struct ethtool_stringsets *dst = yarg->data;
+       unsigned int n_stringset = 0;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+       int i;
+
+       parg.ys = yarg->ys;
+
+       if (dst->stringset)
+               return ynl_error_parse(yarg, "attribute already present (stringsets.stringset)");
+
+       mnl_attr_for_each_nested(attr, nested) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_STRINGSETS_STRINGSET) {
+                       n_stringset++;
+               }
+       }
+
+       if (n_stringset) {
+               dst->stringset = calloc(n_stringset, sizeof(*dst->stringset));
+               dst->n_stringset = n_stringset;
+               i = 0;
+               parg.rsp_policy = &ethtool_stringset_nest;
+               mnl_attr_for_each_nested(attr, nested) {
+                       if (mnl_attr_get_type(attr) == ETHTOOL_A_STRINGSETS_STRINGSET) {
+                               parg.data = &dst->stringset[i];
+                               if (ethtool_stringset_parse(&parg, attr))
+                                       return MNL_CB_ERROR;
+                               i++;
+                       }
+               }
+       }
+
+       return 0;
+}
+
+void ethtool_tunnel_udp_free(struct ethtool_tunnel_udp *obj)
+{
+       ethtool_tunnel_udp_table_free(&obj->table);
+}
+
+int ethtool_tunnel_udp_parse(struct ynl_parse_arg *yarg,
+                            const struct nlattr *nested)
+{
+       struct ethtool_tunnel_udp *dst = yarg->data;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+
+       parg.ys = yarg->ys;
+
+       mnl_attr_for_each_nested(attr, nested) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_TUNNEL_UDP_TABLE) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.table = 1;
+
+                       parg.rsp_policy = &ethtool_tunnel_udp_table_nest;
+                       parg.data = &dst->table;
+                       if (ethtool_tunnel_udp_table_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               }
+       }
+
+       return 0;
+}
+
+/* ============== ETHTOOL_MSG_STRSET_GET ============== */
+/* ETHTOOL_MSG_STRSET_GET - do */
+void ethtool_strset_get_req_free(struct ethtool_strset_get_req *req)
+{
+       ethtool_header_free(&req->header);
+       ethtool_stringsets_free(&req->stringsets);
+       free(req);
+}
+
+void ethtool_strset_get_rsp_free(struct ethtool_strset_get_rsp *rsp)
+{
+       ethtool_header_free(&rsp->header);
+       ethtool_stringsets_free(&rsp->stringsets);
+       free(rsp);
+}
+
+int ethtool_strset_get_rsp_parse(const struct nlmsghdr *nlh, void *data)
+{
+       struct ethtool_strset_get_rsp *dst;
+       struct ynl_parse_arg *yarg = data;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+
+       dst = yarg->data;
+       parg.ys = yarg->ys;
+
+       mnl_attr_for_each(attr, nlh, sizeof(struct genlmsghdr)) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_STRSET_HEADER) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.header = 1;
+
+                       parg.rsp_policy = &ethtool_header_nest;
+                       parg.data = &dst->header;
+                       if (ethtool_header_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_STRSET_STRINGSETS) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.stringsets = 1;
+
+                       parg.rsp_policy = &ethtool_stringsets_nest;
+                       parg.data = &dst->stringsets;
+                       if (ethtool_stringsets_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               }
+       }
+
+       return MNL_CB_OK;
+}
+
+struct ethtool_strset_get_rsp *
+ethtool_strset_get(struct ynl_sock *ys, struct ethtool_strset_get_req *req)
+{
+       struct ynl_req_state yrs = { .yarg = { .ys = ys, }, };
+       struct ethtool_strset_get_rsp *rsp;
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_STRSET_GET, 1);
+       ys->req_policy = &ethtool_strset_nest;
+       yrs.yarg.rsp_policy = &ethtool_strset_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_STRSET_HEADER, &req->header);
+       if (req->_present.stringsets)
+               ethtool_stringsets_put(nlh, ETHTOOL_A_STRSET_STRINGSETS, &req->stringsets);
+       if (req->_present.counts_only)
+               mnl_attr_put(nlh, ETHTOOL_A_STRSET_COUNTS_ONLY, 0, NULL);
+
+       rsp = calloc(1, sizeof(*rsp));
+       yrs.yarg.data = rsp;
+       yrs.cb = ethtool_strset_get_rsp_parse;
+       yrs.rsp_cmd = ETHTOOL_MSG_STRSET_GET;
+
+       err = ynl_exec(ys, nlh, &yrs);
+       if (err < 0)
+               goto err_free;
+
+       return rsp;
+
+err_free:
+       ethtool_strset_get_rsp_free(rsp);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_STRSET_GET - dump */
+void ethtool_strset_get_list_free(struct ethtool_strset_get_list *rsp)
+{
+       struct ethtool_strset_get_list *next = rsp;
+
+       while ((void *)next != YNL_LIST_END) {
+               rsp = next;
+               next = rsp->next;
+
+               ethtool_header_free(&rsp->obj.header);
+               ethtool_stringsets_free(&rsp->obj.stringsets);
+               free(rsp);
+       }
+}
+
+struct ethtool_strset_get_list *
+ethtool_strset_get_dump(struct ynl_sock *ys,
+                       struct ethtool_strset_get_req_dump *req)
+{
+       struct ynl_dump_state yds = {};
+       struct nlmsghdr *nlh;
+       int err;
+
+       yds.ys = ys;
+       yds.alloc_sz = sizeof(struct ethtool_strset_get_list);
+       yds.cb = ethtool_strset_get_rsp_parse;
+       yds.rsp_cmd = ETHTOOL_MSG_STRSET_GET;
+       yds.rsp_policy = &ethtool_strset_nest;
+
+       nlh = ynl_gemsg_start_dump(ys, ys->family_id, ETHTOOL_MSG_STRSET_GET, 1);
+       ys->req_policy = &ethtool_strset_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_STRSET_HEADER, &req->header);
+       if (req->_present.stringsets)
+               ethtool_stringsets_put(nlh, ETHTOOL_A_STRSET_STRINGSETS, &req->stringsets);
+       if (req->_present.counts_only)
+               mnl_attr_put(nlh, ETHTOOL_A_STRSET_COUNTS_ONLY, 0, NULL);
+
+       err = ynl_exec_dump(ys, nlh, &yds);
+       if (err < 0)
+               goto free_list;
+
+       return yds.first;
+
+free_list:
+       ethtool_strset_get_list_free(yds.first);
+       return NULL;
+}
+
+/* ============== ETHTOOL_MSG_LINKINFO_GET ============== */
+/* ETHTOOL_MSG_LINKINFO_GET - do */
+void ethtool_linkinfo_get_req_free(struct ethtool_linkinfo_get_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+void ethtool_linkinfo_get_rsp_free(struct ethtool_linkinfo_get_rsp *rsp)
+{
+       ethtool_header_free(&rsp->header);
+       free(rsp);
+}
+
+int ethtool_linkinfo_get_rsp_parse(const struct nlmsghdr *nlh, void *data)
+{
+       struct ethtool_linkinfo_get_rsp *dst;
+       struct ynl_parse_arg *yarg = data;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+
+       dst = yarg->data;
+       parg.ys = yarg->ys;
+
+       mnl_attr_for_each(attr, nlh, sizeof(struct genlmsghdr)) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_LINKINFO_HEADER) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.header = 1;
+
+                       parg.rsp_policy = &ethtool_header_nest;
+                       parg.data = &dst->header;
+                       if (ethtool_header_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_LINKINFO_PORT) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.port = 1;
+                       dst->port = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_LINKINFO_PHYADDR) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.phyaddr = 1;
+                       dst->phyaddr = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_LINKINFO_TP_MDIX) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.tp_mdix = 1;
+                       dst->tp_mdix = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_LINKINFO_TP_MDIX_CTRL) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.tp_mdix_ctrl = 1;
+                       dst->tp_mdix_ctrl = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_LINKINFO_TRANSCEIVER) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.transceiver = 1;
+                       dst->transceiver = mnl_attr_get_u8(attr);
+               }
+       }
+
+       return MNL_CB_OK;
+}
+
+struct ethtool_linkinfo_get_rsp *
+ethtool_linkinfo_get(struct ynl_sock *ys, struct ethtool_linkinfo_get_req *req)
+{
+       struct ynl_req_state yrs = { .yarg = { .ys = ys, }, };
+       struct ethtool_linkinfo_get_rsp *rsp;
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_LINKINFO_GET, 1);
+       ys->req_policy = &ethtool_linkinfo_nest;
+       yrs.yarg.rsp_policy = &ethtool_linkinfo_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_LINKINFO_HEADER, &req->header);
+
+       rsp = calloc(1, sizeof(*rsp));
+       yrs.yarg.data = rsp;
+       yrs.cb = ethtool_linkinfo_get_rsp_parse;
+       yrs.rsp_cmd = ETHTOOL_MSG_LINKINFO_GET;
+
+       err = ynl_exec(ys, nlh, &yrs);
+       if (err < 0)
+               goto err_free;
+
+       return rsp;
+
+err_free:
+       ethtool_linkinfo_get_rsp_free(rsp);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_LINKINFO_GET - dump */
+void ethtool_linkinfo_get_list_free(struct ethtool_linkinfo_get_list *rsp)
+{
+       struct ethtool_linkinfo_get_list *next = rsp;
+
+       while ((void *)next != YNL_LIST_END) {
+               rsp = next;
+               next = rsp->next;
+
+               ethtool_header_free(&rsp->obj.header);
+               free(rsp);
+       }
+}
+
+struct ethtool_linkinfo_get_list *
+ethtool_linkinfo_get_dump(struct ynl_sock *ys,
+                         struct ethtool_linkinfo_get_req_dump *req)
+{
+       struct ynl_dump_state yds = {};
+       struct nlmsghdr *nlh;
+       int err;
+
+       yds.ys = ys;
+       yds.alloc_sz = sizeof(struct ethtool_linkinfo_get_list);
+       yds.cb = ethtool_linkinfo_get_rsp_parse;
+       yds.rsp_cmd = ETHTOOL_MSG_LINKINFO_GET;
+       yds.rsp_policy = &ethtool_linkinfo_nest;
+
+       nlh = ynl_gemsg_start_dump(ys, ys->family_id, ETHTOOL_MSG_LINKINFO_GET, 1);
+       ys->req_policy = &ethtool_linkinfo_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_LINKINFO_HEADER, &req->header);
+
+       err = ynl_exec_dump(ys, nlh, &yds);
+       if (err < 0)
+               goto free_list;
+
+       return yds.first;
+
+free_list:
+       ethtool_linkinfo_get_list_free(yds.first);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_LINKINFO_GET - notify */
+void ethtool_linkinfo_get_ntf_free(struct ethtool_linkinfo_get_ntf *rsp)
+{
+       ethtool_header_free(&rsp->obj.header);
+       free(rsp);
+}
+
+/* ============== ETHTOOL_MSG_LINKINFO_SET ============== */
+/* ETHTOOL_MSG_LINKINFO_SET - do */
+void ethtool_linkinfo_set_req_free(struct ethtool_linkinfo_set_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+int ethtool_linkinfo_set(struct ynl_sock *ys,
+                        struct ethtool_linkinfo_set_req *req)
+{
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_LINKINFO_SET, 1);
+       ys->req_policy = &ethtool_linkinfo_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_LINKINFO_HEADER, &req->header);
+       if (req->_present.port)
+               mnl_attr_put_u8(nlh, ETHTOOL_A_LINKINFO_PORT, req->port);
+       if (req->_present.phyaddr)
+               mnl_attr_put_u8(nlh, ETHTOOL_A_LINKINFO_PHYADDR, req->phyaddr);
+       if (req->_present.tp_mdix)
+               mnl_attr_put_u8(nlh, ETHTOOL_A_LINKINFO_TP_MDIX, req->tp_mdix);
+       if (req->_present.tp_mdix_ctrl)
+               mnl_attr_put_u8(nlh, ETHTOOL_A_LINKINFO_TP_MDIX_CTRL, req->tp_mdix_ctrl);
+       if (req->_present.transceiver)
+               mnl_attr_put_u8(nlh, ETHTOOL_A_LINKINFO_TRANSCEIVER, req->transceiver);
+
+       err = ynl_exec(ys, nlh, NULL);
+       if (err < 0)
+               return -1;
+
+       return 0;
+}
+
+/* ============== ETHTOOL_MSG_LINKMODES_GET ============== */
+/* ETHTOOL_MSG_LINKMODES_GET - do */
+void ethtool_linkmodes_get_req_free(struct ethtool_linkmodes_get_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+void ethtool_linkmodes_get_rsp_free(struct ethtool_linkmodes_get_rsp *rsp)
+{
+       ethtool_header_free(&rsp->header);
+       ethtool_bitset_free(&rsp->ours);
+       ethtool_bitset_free(&rsp->peer);
+       free(rsp);
+}
+
+int ethtool_linkmodes_get_rsp_parse(const struct nlmsghdr *nlh, void *data)
+{
+       struct ethtool_linkmodes_get_rsp *dst;
+       struct ynl_parse_arg *yarg = data;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+
+       dst = yarg->data;
+       parg.ys = yarg->ys;
+
+       mnl_attr_for_each(attr, nlh, sizeof(struct genlmsghdr)) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_LINKMODES_HEADER) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.header = 1;
+
+                       parg.rsp_policy = &ethtool_header_nest;
+                       parg.data = &dst->header;
+                       if (ethtool_header_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_LINKMODES_AUTONEG) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.autoneg = 1;
+                       dst->autoneg = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_LINKMODES_OURS) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.ours = 1;
+
+                       parg.rsp_policy = &ethtool_bitset_nest;
+                       parg.data = &dst->ours;
+                       if (ethtool_bitset_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_LINKMODES_PEER) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.peer = 1;
+
+                       parg.rsp_policy = &ethtool_bitset_nest;
+                       parg.data = &dst->peer;
+                       if (ethtool_bitset_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_LINKMODES_SPEED) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.speed = 1;
+                       dst->speed = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_LINKMODES_DUPLEX) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.duplex = 1;
+                       dst->duplex = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_LINKMODES_MASTER_SLAVE_CFG) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.master_slave_cfg = 1;
+                       dst->master_slave_cfg = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_LINKMODES_MASTER_SLAVE_STATE) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.master_slave_state = 1;
+                       dst->master_slave_state = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_LINKMODES_LANES) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.lanes = 1;
+                       dst->lanes = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_LINKMODES_RATE_MATCHING) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.rate_matching = 1;
+                       dst->rate_matching = mnl_attr_get_u8(attr);
+               }
+       }
+
+       return MNL_CB_OK;
+}
+
+struct ethtool_linkmodes_get_rsp *
+ethtool_linkmodes_get(struct ynl_sock *ys,
+                     struct ethtool_linkmodes_get_req *req)
+{
+       struct ynl_req_state yrs = { .yarg = { .ys = ys, }, };
+       struct ethtool_linkmodes_get_rsp *rsp;
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_LINKMODES_GET, 1);
+       ys->req_policy = &ethtool_linkmodes_nest;
+       yrs.yarg.rsp_policy = &ethtool_linkmodes_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_LINKMODES_HEADER, &req->header);
+
+       rsp = calloc(1, sizeof(*rsp));
+       yrs.yarg.data = rsp;
+       yrs.cb = ethtool_linkmodes_get_rsp_parse;
+       yrs.rsp_cmd = ETHTOOL_MSG_LINKMODES_GET;
+
+       err = ynl_exec(ys, nlh, &yrs);
+       if (err < 0)
+               goto err_free;
+
+       return rsp;
+
+err_free:
+       ethtool_linkmodes_get_rsp_free(rsp);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_LINKMODES_GET - dump */
+void ethtool_linkmodes_get_list_free(struct ethtool_linkmodes_get_list *rsp)
+{
+       struct ethtool_linkmodes_get_list *next = rsp;
+
+       while ((void *)next != YNL_LIST_END) {
+               rsp = next;
+               next = rsp->next;
+
+               ethtool_header_free(&rsp->obj.header);
+               ethtool_bitset_free(&rsp->obj.ours);
+               ethtool_bitset_free(&rsp->obj.peer);
+               free(rsp);
+       }
+}
+
+struct ethtool_linkmodes_get_list *
+ethtool_linkmodes_get_dump(struct ynl_sock *ys,
+                          struct ethtool_linkmodes_get_req_dump *req)
+{
+       struct ynl_dump_state yds = {};
+       struct nlmsghdr *nlh;
+       int err;
+
+       yds.ys = ys;
+       yds.alloc_sz = sizeof(struct ethtool_linkmodes_get_list);
+       yds.cb = ethtool_linkmodes_get_rsp_parse;
+       yds.rsp_cmd = ETHTOOL_MSG_LINKMODES_GET;
+       yds.rsp_policy = &ethtool_linkmodes_nest;
+
+       nlh = ynl_gemsg_start_dump(ys, ys->family_id, ETHTOOL_MSG_LINKMODES_GET, 1);
+       ys->req_policy = &ethtool_linkmodes_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_LINKMODES_HEADER, &req->header);
+
+       err = ynl_exec_dump(ys, nlh, &yds);
+       if (err < 0)
+               goto free_list;
+
+       return yds.first;
+
+free_list:
+       ethtool_linkmodes_get_list_free(yds.first);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_LINKMODES_GET - notify */
+void ethtool_linkmodes_get_ntf_free(struct ethtool_linkmodes_get_ntf *rsp)
+{
+       ethtool_header_free(&rsp->obj.header);
+       ethtool_bitset_free(&rsp->obj.ours);
+       ethtool_bitset_free(&rsp->obj.peer);
+       free(rsp);
+}
+
+/* ============== ETHTOOL_MSG_LINKMODES_SET ============== */
+/* ETHTOOL_MSG_LINKMODES_SET - do */
+void ethtool_linkmodes_set_req_free(struct ethtool_linkmodes_set_req *req)
+{
+       ethtool_header_free(&req->header);
+       ethtool_bitset_free(&req->ours);
+       ethtool_bitset_free(&req->peer);
+       free(req);
+}
+
+int ethtool_linkmodes_set(struct ynl_sock *ys,
+                         struct ethtool_linkmodes_set_req *req)
+{
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_LINKMODES_SET, 1);
+       ys->req_policy = &ethtool_linkmodes_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_LINKMODES_HEADER, &req->header);
+       if (req->_present.autoneg)
+               mnl_attr_put_u8(nlh, ETHTOOL_A_LINKMODES_AUTONEG, req->autoneg);
+       if (req->_present.ours)
+               ethtool_bitset_put(nlh, ETHTOOL_A_LINKMODES_OURS, &req->ours);
+       if (req->_present.peer)
+               ethtool_bitset_put(nlh, ETHTOOL_A_LINKMODES_PEER, &req->peer);
+       if (req->_present.speed)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_LINKMODES_SPEED, req->speed);
+       if (req->_present.duplex)
+               mnl_attr_put_u8(nlh, ETHTOOL_A_LINKMODES_DUPLEX, req->duplex);
+       if (req->_present.master_slave_cfg)
+               mnl_attr_put_u8(nlh, ETHTOOL_A_LINKMODES_MASTER_SLAVE_CFG, req->master_slave_cfg);
+       if (req->_present.master_slave_state)
+               mnl_attr_put_u8(nlh, ETHTOOL_A_LINKMODES_MASTER_SLAVE_STATE, req->master_slave_state);
+       if (req->_present.lanes)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_LINKMODES_LANES, req->lanes);
+       if (req->_present.rate_matching)
+               mnl_attr_put_u8(nlh, ETHTOOL_A_LINKMODES_RATE_MATCHING, req->rate_matching);
+
+       err = ynl_exec(ys, nlh, NULL);
+       if (err < 0)
+               return -1;
+
+       return 0;
+}
+
+/* ============== ETHTOOL_MSG_LINKSTATE_GET ============== */
+/* ETHTOOL_MSG_LINKSTATE_GET - do */
+void ethtool_linkstate_get_req_free(struct ethtool_linkstate_get_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+void ethtool_linkstate_get_rsp_free(struct ethtool_linkstate_get_rsp *rsp)
+{
+       ethtool_header_free(&rsp->header);
+       free(rsp);
+}
+
+int ethtool_linkstate_get_rsp_parse(const struct nlmsghdr *nlh, void *data)
+{
+       struct ethtool_linkstate_get_rsp *dst;
+       struct ynl_parse_arg *yarg = data;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+
+       dst = yarg->data;
+       parg.ys = yarg->ys;
+
+       mnl_attr_for_each(attr, nlh, sizeof(struct genlmsghdr)) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_LINKSTATE_HEADER) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.header = 1;
+
+                       parg.rsp_policy = &ethtool_header_nest;
+                       parg.data = &dst->header;
+                       if (ethtool_header_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_LINKSTATE_LINK) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.link = 1;
+                       dst->link = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_LINKSTATE_SQI) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.sqi = 1;
+                       dst->sqi = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_LINKSTATE_SQI_MAX) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.sqi_max = 1;
+                       dst->sqi_max = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_LINKSTATE_EXT_STATE) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.ext_state = 1;
+                       dst->ext_state = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_LINKSTATE_EXT_SUBSTATE) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.ext_substate = 1;
+                       dst->ext_substate = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_LINKSTATE_EXT_DOWN_CNT) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.ext_down_cnt = 1;
+                       dst->ext_down_cnt = mnl_attr_get_u32(attr);
+               }
+       }
+
+       return MNL_CB_OK;
+}
+
+struct ethtool_linkstate_get_rsp *
+ethtool_linkstate_get(struct ynl_sock *ys,
+                     struct ethtool_linkstate_get_req *req)
+{
+       struct ynl_req_state yrs = { .yarg = { .ys = ys, }, };
+       struct ethtool_linkstate_get_rsp *rsp;
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_LINKSTATE_GET, 1);
+       ys->req_policy = &ethtool_linkstate_nest;
+       yrs.yarg.rsp_policy = &ethtool_linkstate_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_LINKSTATE_HEADER, &req->header);
+
+       rsp = calloc(1, sizeof(*rsp));
+       yrs.yarg.data = rsp;
+       yrs.cb = ethtool_linkstate_get_rsp_parse;
+       yrs.rsp_cmd = ETHTOOL_MSG_LINKSTATE_GET;
+
+       err = ynl_exec(ys, nlh, &yrs);
+       if (err < 0)
+               goto err_free;
+
+       return rsp;
+
+err_free:
+       ethtool_linkstate_get_rsp_free(rsp);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_LINKSTATE_GET - dump */
+void ethtool_linkstate_get_list_free(struct ethtool_linkstate_get_list *rsp)
+{
+       struct ethtool_linkstate_get_list *next = rsp;
+
+       while ((void *)next != YNL_LIST_END) {
+               rsp = next;
+               next = rsp->next;
+
+               ethtool_header_free(&rsp->obj.header);
+               free(rsp);
+       }
+}
+
+struct ethtool_linkstate_get_list *
+ethtool_linkstate_get_dump(struct ynl_sock *ys,
+                          struct ethtool_linkstate_get_req_dump *req)
+{
+       struct ynl_dump_state yds = {};
+       struct nlmsghdr *nlh;
+       int err;
+
+       yds.ys = ys;
+       yds.alloc_sz = sizeof(struct ethtool_linkstate_get_list);
+       yds.cb = ethtool_linkstate_get_rsp_parse;
+       yds.rsp_cmd = ETHTOOL_MSG_LINKSTATE_GET;
+       yds.rsp_policy = &ethtool_linkstate_nest;
+
+       nlh = ynl_gemsg_start_dump(ys, ys->family_id, ETHTOOL_MSG_LINKSTATE_GET, 1);
+       ys->req_policy = &ethtool_linkstate_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_LINKSTATE_HEADER, &req->header);
+
+       err = ynl_exec_dump(ys, nlh, &yds);
+       if (err < 0)
+               goto free_list;
+
+       return yds.first;
+
+free_list:
+       ethtool_linkstate_get_list_free(yds.first);
+       return NULL;
+}
+
+/* ============== ETHTOOL_MSG_DEBUG_GET ============== */
+/* ETHTOOL_MSG_DEBUG_GET - do */
+void ethtool_debug_get_req_free(struct ethtool_debug_get_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+void ethtool_debug_get_rsp_free(struct ethtool_debug_get_rsp *rsp)
+{
+       ethtool_header_free(&rsp->header);
+       ethtool_bitset_free(&rsp->msgmask);
+       free(rsp);
+}
+
+int ethtool_debug_get_rsp_parse(const struct nlmsghdr *nlh, void *data)
+{
+       struct ethtool_debug_get_rsp *dst;
+       struct ynl_parse_arg *yarg = data;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+
+       dst = yarg->data;
+       parg.ys = yarg->ys;
+
+       mnl_attr_for_each(attr, nlh, sizeof(struct genlmsghdr)) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_DEBUG_HEADER) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.header = 1;
+
+                       parg.rsp_policy = &ethtool_header_nest;
+                       parg.data = &dst->header;
+                       if (ethtool_header_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_DEBUG_MSGMASK) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.msgmask = 1;
+
+                       parg.rsp_policy = &ethtool_bitset_nest;
+                       parg.data = &dst->msgmask;
+                       if (ethtool_bitset_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               }
+       }
+
+       return MNL_CB_OK;
+}
+
+struct ethtool_debug_get_rsp *
+ethtool_debug_get(struct ynl_sock *ys, struct ethtool_debug_get_req *req)
+{
+       struct ynl_req_state yrs = { .yarg = { .ys = ys, }, };
+       struct ethtool_debug_get_rsp *rsp;
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_DEBUG_GET, 1);
+       ys->req_policy = &ethtool_debug_nest;
+       yrs.yarg.rsp_policy = &ethtool_debug_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_DEBUG_HEADER, &req->header);
+
+       rsp = calloc(1, sizeof(*rsp));
+       yrs.yarg.data = rsp;
+       yrs.cb = ethtool_debug_get_rsp_parse;
+       yrs.rsp_cmd = ETHTOOL_MSG_DEBUG_GET;
+
+       err = ynl_exec(ys, nlh, &yrs);
+       if (err < 0)
+               goto err_free;
+
+       return rsp;
+
+err_free:
+       ethtool_debug_get_rsp_free(rsp);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_DEBUG_GET - dump */
+void ethtool_debug_get_list_free(struct ethtool_debug_get_list *rsp)
+{
+       struct ethtool_debug_get_list *next = rsp;
+
+       while ((void *)next != YNL_LIST_END) {
+               rsp = next;
+               next = rsp->next;
+
+               ethtool_header_free(&rsp->obj.header);
+               ethtool_bitset_free(&rsp->obj.msgmask);
+               free(rsp);
+       }
+}
+
+struct ethtool_debug_get_list *
+ethtool_debug_get_dump(struct ynl_sock *ys,
+                      struct ethtool_debug_get_req_dump *req)
+{
+       struct ynl_dump_state yds = {};
+       struct nlmsghdr *nlh;
+       int err;
+
+       yds.ys = ys;
+       yds.alloc_sz = sizeof(struct ethtool_debug_get_list);
+       yds.cb = ethtool_debug_get_rsp_parse;
+       yds.rsp_cmd = ETHTOOL_MSG_DEBUG_GET;
+       yds.rsp_policy = &ethtool_debug_nest;
+
+       nlh = ynl_gemsg_start_dump(ys, ys->family_id, ETHTOOL_MSG_DEBUG_GET, 1);
+       ys->req_policy = &ethtool_debug_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_DEBUG_HEADER, &req->header);
+
+       err = ynl_exec_dump(ys, nlh, &yds);
+       if (err < 0)
+               goto free_list;
+
+       return yds.first;
+
+free_list:
+       ethtool_debug_get_list_free(yds.first);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_DEBUG_GET - notify */
+void ethtool_debug_get_ntf_free(struct ethtool_debug_get_ntf *rsp)
+{
+       ethtool_header_free(&rsp->obj.header);
+       ethtool_bitset_free(&rsp->obj.msgmask);
+       free(rsp);
+}
+
+/* ============== ETHTOOL_MSG_DEBUG_SET ============== */
+/* ETHTOOL_MSG_DEBUG_SET - do */
+void ethtool_debug_set_req_free(struct ethtool_debug_set_req *req)
+{
+       ethtool_header_free(&req->header);
+       ethtool_bitset_free(&req->msgmask);
+       free(req);
+}
+
+int ethtool_debug_set(struct ynl_sock *ys, struct ethtool_debug_set_req *req)
+{
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_DEBUG_SET, 1);
+       ys->req_policy = &ethtool_debug_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_DEBUG_HEADER, &req->header);
+       if (req->_present.msgmask)
+               ethtool_bitset_put(nlh, ETHTOOL_A_DEBUG_MSGMASK, &req->msgmask);
+
+       err = ynl_exec(ys, nlh, NULL);
+       if (err < 0)
+               return -1;
+
+       return 0;
+}
+
+/* ============== ETHTOOL_MSG_WOL_GET ============== */
+/* ETHTOOL_MSG_WOL_GET - do */
+void ethtool_wol_get_req_free(struct ethtool_wol_get_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+void ethtool_wol_get_rsp_free(struct ethtool_wol_get_rsp *rsp)
+{
+       ethtool_header_free(&rsp->header);
+       ethtool_bitset_free(&rsp->modes);
+       free(rsp->sopass);
+       free(rsp);
+}
+
+int ethtool_wol_get_rsp_parse(const struct nlmsghdr *nlh, void *data)
+{
+       struct ynl_parse_arg *yarg = data;
+       struct ethtool_wol_get_rsp *dst;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+
+       dst = yarg->data;
+       parg.ys = yarg->ys;
+
+       mnl_attr_for_each(attr, nlh, sizeof(struct genlmsghdr)) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_WOL_HEADER) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.header = 1;
+
+                       parg.rsp_policy = &ethtool_header_nest;
+                       parg.data = &dst->header;
+                       if (ethtool_header_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_WOL_MODES) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.modes = 1;
+
+                       parg.rsp_policy = &ethtool_bitset_nest;
+                       parg.data = &dst->modes;
+                       if (ethtool_bitset_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_WOL_SOPASS) {
+                       unsigned int len;
+
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+
+                       len = mnl_attr_get_payload_len(attr);
+                       dst->_present.sopass_len = len;
+                       dst->sopass = malloc(len);
+                       memcpy(dst->sopass, mnl_attr_get_payload(attr), len);
+               }
+       }
+
+       return MNL_CB_OK;
+}
+
+struct ethtool_wol_get_rsp *
+ethtool_wol_get(struct ynl_sock *ys, struct ethtool_wol_get_req *req)
+{
+       struct ynl_req_state yrs = { .yarg = { .ys = ys, }, };
+       struct ethtool_wol_get_rsp *rsp;
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_WOL_GET, 1);
+       ys->req_policy = &ethtool_wol_nest;
+       yrs.yarg.rsp_policy = &ethtool_wol_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_WOL_HEADER, &req->header);
+
+       rsp = calloc(1, sizeof(*rsp));
+       yrs.yarg.data = rsp;
+       yrs.cb = ethtool_wol_get_rsp_parse;
+       yrs.rsp_cmd = ETHTOOL_MSG_WOL_GET;
+
+       err = ynl_exec(ys, nlh, &yrs);
+       if (err < 0)
+               goto err_free;
+
+       return rsp;
+
+err_free:
+       ethtool_wol_get_rsp_free(rsp);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_WOL_GET - dump */
+void ethtool_wol_get_list_free(struct ethtool_wol_get_list *rsp)
+{
+       struct ethtool_wol_get_list *next = rsp;
+
+       while ((void *)next != YNL_LIST_END) {
+               rsp = next;
+               next = rsp->next;
+
+               ethtool_header_free(&rsp->obj.header);
+               ethtool_bitset_free(&rsp->obj.modes);
+               free(rsp->obj.sopass);
+               free(rsp);
+       }
+}
+
+struct ethtool_wol_get_list *
+ethtool_wol_get_dump(struct ynl_sock *ys, struct ethtool_wol_get_req_dump *req)
+{
+       struct ynl_dump_state yds = {};
+       struct nlmsghdr *nlh;
+       int err;
+
+       yds.ys = ys;
+       yds.alloc_sz = sizeof(struct ethtool_wol_get_list);
+       yds.cb = ethtool_wol_get_rsp_parse;
+       yds.rsp_cmd = ETHTOOL_MSG_WOL_GET;
+       yds.rsp_policy = &ethtool_wol_nest;
+
+       nlh = ynl_gemsg_start_dump(ys, ys->family_id, ETHTOOL_MSG_WOL_GET, 1);
+       ys->req_policy = &ethtool_wol_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_WOL_HEADER, &req->header);
+
+       err = ynl_exec_dump(ys, nlh, &yds);
+       if (err < 0)
+               goto free_list;
+
+       return yds.first;
+
+free_list:
+       ethtool_wol_get_list_free(yds.first);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_WOL_GET - notify */
+void ethtool_wol_get_ntf_free(struct ethtool_wol_get_ntf *rsp)
+{
+       ethtool_header_free(&rsp->obj.header);
+       ethtool_bitset_free(&rsp->obj.modes);
+       free(rsp->obj.sopass);
+       free(rsp);
+}
+
+/* ============== ETHTOOL_MSG_WOL_SET ============== */
+/* ETHTOOL_MSG_WOL_SET - do */
+void ethtool_wol_set_req_free(struct ethtool_wol_set_req *req)
+{
+       ethtool_header_free(&req->header);
+       ethtool_bitset_free(&req->modes);
+       free(req->sopass);
+       free(req);
+}
+
+int ethtool_wol_set(struct ynl_sock *ys, struct ethtool_wol_set_req *req)
+{
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_WOL_SET, 1);
+       ys->req_policy = &ethtool_wol_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_WOL_HEADER, &req->header);
+       if (req->_present.modes)
+               ethtool_bitset_put(nlh, ETHTOOL_A_WOL_MODES, &req->modes);
+       if (req->_present.sopass_len)
+               mnl_attr_put(nlh, ETHTOOL_A_WOL_SOPASS, req->_present.sopass_len, req->sopass);
+
+       err = ynl_exec(ys, nlh, NULL);
+       if (err < 0)
+               return -1;
+
+       return 0;
+}
+
+/* ============== ETHTOOL_MSG_FEATURES_GET ============== */
+/* ETHTOOL_MSG_FEATURES_GET - do */
+void ethtool_features_get_req_free(struct ethtool_features_get_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+void ethtool_features_get_rsp_free(struct ethtool_features_get_rsp *rsp)
+{
+       ethtool_header_free(&rsp->header);
+       ethtool_bitset_free(&rsp->hw);
+       ethtool_bitset_free(&rsp->wanted);
+       ethtool_bitset_free(&rsp->active);
+       ethtool_bitset_free(&rsp->nochange);
+       free(rsp);
+}
+
+int ethtool_features_get_rsp_parse(const struct nlmsghdr *nlh, void *data)
+{
+       struct ethtool_features_get_rsp *dst;
+       struct ynl_parse_arg *yarg = data;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+
+       dst = yarg->data;
+       parg.ys = yarg->ys;
+
+       mnl_attr_for_each(attr, nlh, sizeof(struct genlmsghdr)) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_FEATURES_HEADER) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.header = 1;
+
+                       parg.rsp_policy = &ethtool_header_nest;
+                       parg.data = &dst->header;
+                       if (ethtool_header_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_FEATURES_HW) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.hw = 1;
+
+                       parg.rsp_policy = &ethtool_bitset_nest;
+                       parg.data = &dst->hw;
+                       if (ethtool_bitset_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_FEATURES_WANTED) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.wanted = 1;
+
+                       parg.rsp_policy = &ethtool_bitset_nest;
+                       parg.data = &dst->wanted;
+                       if (ethtool_bitset_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_FEATURES_ACTIVE) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.active = 1;
+
+                       parg.rsp_policy = &ethtool_bitset_nest;
+                       parg.data = &dst->active;
+                       if (ethtool_bitset_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_FEATURES_NOCHANGE) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.nochange = 1;
+
+                       parg.rsp_policy = &ethtool_bitset_nest;
+                       parg.data = &dst->nochange;
+                       if (ethtool_bitset_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               }
+       }
+
+       return MNL_CB_OK;
+}
+
+struct ethtool_features_get_rsp *
+ethtool_features_get(struct ynl_sock *ys, struct ethtool_features_get_req *req)
+{
+       struct ynl_req_state yrs = { .yarg = { .ys = ys, }, };
+       struct ethtool_features_get_rsp *rsp;
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_FEATURES_GET, 1);
+       ys->req_policy = &ethtool_features_nest;
+       yrs.yarg.rsp_policy = &ethtool_features_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_FEATURES_HEADER, &req->header);
+
+       rsp = calloc(1, sizeof(*rsp));
+       yrs.yarg.data = rsp;
+       yrs.cb = ethtool_features_get_rsp_parse;
+       yrs.rsp_cmd = ETHTOOL_MSG_FEATURES_GET;
+
+       err = ynl_exec(ys, nlh, &yrs);
+       if (err < 0)
+               goto err_free;
+
+       return rsp;
+
+err_free:
+       ethtool_features_get_rsp_free(rsp);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_FEATURES_GET - dump */
+void ethtool_features_get_list_free(struct ethtool_features_get_list *rsp)
+{
+       struct ethtool_features_get_list *next = rsp;
+
+       while ((void *)next != YNL_LIST_END) {
+               rsp = next;
+               next = rsp->next;
+
+               ethtool_header_free(&rsp->obj.header);
+               ethtool_bitset_free(&rsp->obj.hw);
+               ethtool_bitset_free(&rsp->obj.wanted);
+               ethtool_bitset_free(&rsp->obj.active);
+               ethtool_bitset_free(&rsp->obj.nochange);
+               free(rsp);
+       }
+}
+
+struct ethtool_features_get_list *
+ethtool_features_get_dump(struct ynl_sock *ys,
+                         struct ethtool_features_get_req_dump *req)
+{
+       struct ynl_dump_state yds = {};
+       struct nlmsghdr *nlh;
+       int err;
+
+       yds.ys = ys;
+       yds.alloc_sz = sizeof(struct ethtool_features_get_list);
+       yds.cb = ethtool_features_get_rsp_parse;
+       yds.rsp_cmd = ETHTOOL_MSG_FEATURES_GET;
+       yds.rsp_policy = &ethtool_features_nest;
+
+       nlh = ynl_gemsg_start_dump(ys, ys->family_id, ETHTOOL_MSG_FEATURES_GET, 1);
+       ys->req_policy = &ethtool_features_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_FEATURES_HEADER, &req->header);
+
+       err = ynl_exec_dump(ys, nlh, &yds);
+       if (err < 0)
+               goto free_list;
+
+       return yds.first;
+
+free_list:
+       ethtool_features_get_list_free(yds.first);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_FEATURES_GET - notify */
+void ethtool_features_get_ntf_free(struct ethtool_features_get_ntf *rsp)
+{
+       ethtool_header_free(&rsp->obj.header);
+       ethtool_bitset_free(&rsp->obj.hw);
+       ethtool_bitset_free(&rsp->obj.wanted);
+       ethtool_bitset_free(&rsp->obj.active);
+       ethtool_bitset_free(&rsp->obj.nochange);
+       free(rsp);
+}
+
+/* ============== ETHTOOL_MSG_FEATURES_SET ============== */
+/* ETHTOOL_MSG_FEATURES_SET - do */
+void ethtool_features_set_req_free(struct ethtool_features_set_req *req)
+{
+       ethtool_header_free(&req->header);
+       ethtool_bitset_free(&req->hw);
+       ethtool_bitset_free(&req->wanted);
+       ethtool_bitset_free(&req->active);
+       ethtool_bitset_free(&req->nochange);
+       free(req);
+}
+
+void ethtool_features_set_rsp_free(struct ethtool_features_set_rsp *rsp)
+{
+       ethtool_header_free(&rsp->header);
+       ethtool_bitset_free(&rsp->hw);
+       ethtool_bitset_free(&rsp->wanted);
+       ethtool_bitset_free(&rsp->active);
+       ethtool_bitset_free(&rsp->nochange);
+       free(rsp);
+}
+
+int ethtool_features_set_rsp_parse(const struct nlmsghdr *nlh, void *data)
+{
+       struct ethtool_features_set_rsp *dst;
+       struct ynl_parse_arg *yarg = data;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+
+       dst = yarg->data;
+       parg.ys = yarg->ys;
+
+       mnl_attr_for_each(attr, nlh, sizeof(struct genlmsghdr)) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_FEATURES_HEADER) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.header = 1;
+
+                       parg.rsp_policy = &ethtool_header_nest;
+                       parg.data = &dst->header;
+                       if (ethtool_header_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_FEATURES_HW) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.hw = 1;
+
+                       parg.rsp_policy = &ethtool_bitset_nest;
+                       parg.data = &dst->hw;
+                       if (ethtool_bitset_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_FEATURES_WANTED) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.wanted = 1;
+
+                       parg.rsp_policy = &ethtool_bitset_nest;
+                       parg.data = &dst->wanted;
+                       if (ethtool_bitset_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_FEATURES_ACTIVE) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.active = 1;
+
+                       parg.rsp_policy = &ethtool_bitset_nest;
+                       parg.data = &dst->active;
+                       if (ethtool_bitset_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_FEATURES_NOCHANGE) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.nochange = 1;
+
+                       parg.rsp_policy = &ethtool_bitset_nest;
+                       parg.data = &dst->nochange;
+                       if (ethtool_bitset_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               }
+       }
+
+       return MNL_CB_OK;
+}
+
+struct ethtool_features_set_rsp *
+ethtool_features_set(struct ynl_sock *ys, struct ethtool_features_set_req *req)
+{
+       struct ynl_req_state yrs = { .yarg = { .ys = ys, }, };
+       struct ethtool_features_set_rsp *rsp;
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_FEATURES_SET, 1);
+       ys->req_policy = &ethtool_features_nest;
+       yrs.yarg.rsp_policy = &ethtool_features_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_FEATURES_HEADER, &req->header);
+       if (req->_present.hw)
+               ethtool_bitset_put(nlh, ETHTOOL_A_FEATURES_HW, &req->hw);
+       if (req->_present.wanted)
+               ethtool_bitset_put(nlh, ETHTOOL_A_FEATURES_WANTED, &req->wanted);
+       if (req->_present.active)
+               ethtool_bitset_put(nlh, ETHTOOL_A_FEATURES_ACTIVE, &req->active);
+       if (req->_present.nochange)
+               ethtool_bitset_put(nlh, ETHTOOL_A_FEATURES_NOCHANGE, &req->nochange);
+
+       rsp = calloc(1, sizeof(*rsp));
+       yrs.yarg.data = rsp;
+       yrs.cb = ethtool_features_set_rsp_parse;
+       yrs.rsp_cmd = ETHTOOL_MSG_FEATURES_SET;
+
+       err = ynl_exec(ys, nlh, &yrs);
+       if (err < 0)
+               goto err_free;
+
+       return rsp;
+
+err_free:
+       ethtool_features_set_rsp_free(rsp);
+       return NULL;
+}
+
+/* ============== ETHTOOL_MSG_PRIVFLAGS_GET ============== */
+/* ETHTOOL_MSG_PRIVFLAGS_GET - do */
+void ethtool_privflags_get_req_free(struct ethtool_privflags_get_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+void ethtool_privflags_get_rsp_free(struct ethtool_privflags_get_rsp *rsp)
+{
+       ethtool_header_free(&rsp->header);
+       ethtool_bitset_free(&rsp->flags);
+       free(rsp);
+}
+
+int ethtool_privflags_get_rsp_parse(const struct nlmsghdr *nlh, void *data)
+{
+       struct ethtool_privflags_get_rsp *dst;
+       struct ynl_parse_arg *yarg = data;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+
+       dst = yarg->data;
+       parg.ys = yarg->ys;
+
+       mnl_attr_for_each(attr, nlh, sizeof(struct genlmsghdr)) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_PRIVFLAGS_HEADER) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.header = 1;
+
+                       parg.rsp_policy = &ethtool_header_nest;
+                       parg.data = &dst->header;
+                       if (ethtool_header_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_PRIVFLAGS_FLAGS) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.flags = 1;
+
+                       parg.rsp_policy = &ethtool_bitset_nest;
+                       parg.data = &dst->flags;
+                       if (ethtool_bitset_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               }
+       }
+
+       return MNL_CB_OK;
+}
+
+struct ethtool_privflags_get_rsp *
+ethtool_privflags_get(struct ynl_sock *ys,
+                     struct ethtool_privflags_get_req *req)
+{
+       struct ynl_req_state yrs = { .yarg = { .ys = ys, }, };
+       struct ethtool_privflags_get_rsp *rsp;
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_PRIVFLAGS_GET, 1);
+       ys->req_policy = &ethtool_privflags_nest;
+       yrs.yarg.rsp_policy = &ethtool_privflags_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_PRIVFLAGS_HEADER, &req->header);
+
+       rsp = calloc(1, sizeof(*rsp));
+       yrs.yarg.data = rsp;
+       yrs.cb = ethtool_privflags_get_rsp_parse;
+       yrs.rsp_cmd = 14;
+
+       err = ynl_exec(ys, nlh, &yrs);
+       if (err < 0)
+               goto err_free;
+
+       return rsp;
+
+err_free:
+       ethtool_privflags_get_rsp_free(rsp);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_PRIVFLAGS_GET - dump */
+void ethtool_privflags_get_list_free(struct ethtool_privflags_get_list *rsp)
+{
+       struct ethtool_privflags_get_list *next = rsp;
+
+       while ((void *)next != YNL_LIST_END) {
+               rsp = next;
+               next = rsp->next;
+
+               ethtool_header_free(&rsp->obj.header);
+               ethtool_bitset_free(&rsp->obj.flags);
+               free(rsp);
+       }
+}
+
+struct ethtool_privflags_get_list *
+ethtool_privflags_get_dump(struct ynl_sock *ys,
+                          struct ethtool_privflags_get_req_dump *req)
+{
+       struct ynl_dump_state yds = {};
+       struct nlmsghdr *nlh;
+       int err;
+
+       yds.ys = ys;
+       yds.alloc_sz = sizeof(struct ethtool_privflags_get_list);
+       yds.cb = ethtool_privflags_get_rsp_parse;
+       yds.rsp_cmd = 14;
+       yds.rsp_policy = &ethtool_privflags_nest;
+
+       nlh = ynl_gemsg_start_dump(ys, ys->family_id, ETHTOOL_MSG_PRIVFLAGS_GET, 1);
+       ys->req_policy = &ethtool_privflags_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_PRIVFLAGS_HEADER, &req->header);
+
+       err = ynl_exec_dump(ys, nlh, &yds);
+       if (err < 0)
+               goto free_list;
+
+       return yds.first;
+
+free_list:
+       ethtool_privflags_get_list_free(yds.first);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_PRIVFLAGS_GET - notify */
+void ethtool_privflags_get_ntf_free(struct ethtool_privflags_get_ntf *rsp)
+{
+       ethtool_header_free(&rsp->obj.header);
+       ethtool_bitset_free(&rsp->obj.flags);
+       free(rsp);
+}
+
+/* ============== ETHTOOL_MSG_PRIVFLAGS_SET ============== */
+/* ETHTOOL_MSG_PRIVFLAGS_SET - do */
+void ethtool_privflags_set_req_free(struct ethtool_privflags_set_req *req)
+{
+       ethtool_header_free(&req->header);
+       ethtool_bitset_free(&req->flags);
+       free(req);
+}
+
+int ethtool_privflags_set(struct ynl_sock *ys,
+                         struct ethtool_privflags_set_req *req)
+{
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_PRIVFLAGS_SET, 1);
+       ys->req_policy = &ethtool_privflags_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_PRIVFLAGS_HEADER, &req->header);
+       if (req->_present.flags)
+               ethtool_bitset_put(nlh, ETHTOOL_A_PRIVFLAGS_FLAGS, &req->flags);
+
+       err = ynl_exec(ys, nlh, NULL);
+       if (err < 0)
+               return -1;
+
+       return 0;
+}
+
+/* ============== ETHTOOL_MSG_RINGS_GET ============== */
+/* ETHTOOL_MSG_RINGS_GET - do */
+void ethtool_rings_get_req_free(struct ethtool_rings_get_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+void ethtool_rings_get_rsp_free(struct ethtool_rings_get_rsp *rsp)
+{
+       ethtool_header_free(&rsp->header);
+       free(rsp);
+}
+
+int ethtool_rings_get_rsp_parse(const struct nlmsghdr *nlh, void *data)
+{
+       struct ethtool_rings_get_rsp *dst;
+       struct ynl_parse_arg *yarg = data;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+
+       dst = yarg->data;
+       parg.ys = yarg->ys;
+
+       mnl_attr_for_each(attr, nlh, sizeof(struct genlmsghdr)) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_RINGS_HEADER) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.header = 1;
+
+                       parg.rsp_policy = &ethtool_header_nest;
+                       parg.data = &dst->header;
+                       if (ethtool_header_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_RINGS_RX_MAX) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.rx_max = 1;
+                       dst->rx_max = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_RINGS_RX_MINI_MAX) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.rx_mini_max = 1;
+                       dst->rx_mini_max = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_RINGS_RX_JUMBO_MAX) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.rx_jumbo_max = 1;
+                       dst->rx_jumbo_max = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_RINGS_TX_MAX) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.tx_max = 1;
+                       dst->tx_max = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_RINGS_RX) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.rx = 1;
+                       dst->rx = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_RINGS_RX_MINI) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.rx_mini = 1;
+                       dst->rx_mini = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_RINGS_RX_JUMBO) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.rx_jumbo = 1;
+                       dst->rx_jumbo = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_RINGS_TX) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.tx = 1;
+                       dst->tx = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_RINGS_RX_BUF_LEN) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.rx_buf_len = 1;
+                       dst->rx_buf_len = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_RINGS_TCP_DATA_SPLIT) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.tcp_data_split = 1;
+                       dst->tcp_data_split = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_RINGS_CQE_SIZE) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.cqe_size = 1;
+                       dst->cqe_size = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_RINGS_TX_PUSH) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.tx_push = 1;
+                       dst->tx_push = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_RINGS_RX_PUSH) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.rx_push = 1;
+                       dst->rx_push = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_RINGS_TX_PUSH_BUF_LEN) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.tx_push_buf_len = 1;
+                       dst->tx_push_buf_len = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_RINGS_TX_PUSH_BUF_LEN_MAX) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.tx_push_buf_len_max = 1;
+                       dst->tx_push_buf_len_max = mnl_attr_get_u32(attr);
+               }
+       }
+
+       return MNL_CB_OK;
+}
+
+struct ethtool_rings_get_rsp *
+ethtool_rings_get(struct ynl_sock *ys, struct ethtool_rings_get_req *req)
+{
+       struct ynl_req_state yrs = { .yarg = { .ys = ys, }, };
+       struct ethtool_rings_get_rsp *rsp;
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_RINGS_GET, 1);
+       ys->req_policy = &ethtool_rings_nest;
+       yrs.yarg.rsp_policy = &ethtool_rings_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_RINGS_HEADER, &req->header);
+
+       rsp = calloc(1, sizeof(*rsp));
+       yrs.yarg.data = rsp;
+       yrs.cb = ethtool_rings_get_rsp_parse;
+       yrs.rsp_cmd = 16;
+
+       err = ynl_exec(ys, nlh, &yrs);
+       if (err < 0)
+               goto err_free;
+
+       return rsp;
+
+err_free:
+       ethtool_rings_get_rsp_free(rsp);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_RINGS_GET - dump */
+void ethtool_rings_get_list_free(struct ethtool_rings_get_list *rsp)
+{
+       struct ethtool_rings_get_list *next = rsp;
+
+       while ((void *)next != YNL_LIST_END) {
+               rsp = next;
+               next = rsp->next;
+
+               ethtool_header_free(&rsp->obj.header);
+               free(rsp);
+       }
+}
+
+struct ethtool_rings_get_list *
+ethtool_rings_get_dump(struct ynl_sock *ys,
+                      struct ethtool_rings_get_req_dump *req)
+{
+       struct ynl_dump_state yds = {};
+       struct nlmsghdr *nlh;
+       int err;
+
+       yds.ys = ys;
+       yds.alloc_sz = sizeof(struct ethtool_rings_get_list);
+       yds.cb = ethtool_rings_get_rsp_parse;
+       yds.rsp_cmd = 16;
+       yds.rsp_policy = &ethtool_rings_nest;
+
+       nlh = ynl_gemsg_start_dump(ys, ys->family_id, ETHTOOL_MSG_RINGS_GET, 1);
+       ys->req_policy = &ethtool_rings_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_RINGS_HEADER, &req->header);
+
+       err = ynl_exec_dump(ys, nlh, &yds);
+       if (err < 0)
+               goto free_list;
+
+       return yds.first;
+
+free_list:
+       ethtool_rings_get_list_free(yds.first);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_RINGS_GET - notify */
+void ethtool_rings_get_ntf_free(struct ethtool_rings_get_ntf *rsp)
+{
+       ethtool_header_free(&rsp->obj.header);
+       free(rsp);
+}
+
+/* ============== ETHTOOL_MSG_RINGS_SET ============== */
+/* ETHTOOL_MSG_RINGS_SET - do */
+void ethtool_rings_set_req_free(struct ethtool_rings_set_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+int ethtool_rings_set(struct ynl_sock *ys, struct ethtool_rings_set_req *req)
+{
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_RINGS_SET, 1);
+       ys->req_policy = &ethtool_rings_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_RINGS_HEADER, &req->header);
+       if (req->_present.rx_max)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_RINGS_RX_MAX, req->rx_max);
+       if (req->_present.rx_mini_max)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_RINGS_RX_MINI_MAX, req->rx_mini_max);
+       if (req->_present.rx_jumbo_max)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_RINGS_RX_JUMBO_MAX, req->rx_jumbo_max);
+       if (req->_present.tx_max)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_RINGS_TX_MAX, req->tx_max);
+       if (req->_present.rx)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_RINGS_RX, req->rx);
+       if (req->_present.rx_mini)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_RINGS_RX_MINI, req->rx_mini);
+       if (req->_present.rx_jumbo)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_RINGS_RX_JUMBO, req->rx_jumbo);
+       if (req->_present.tx)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_RINGS_TX, req->tx);
+       if (req->_present.rx_buf_len)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_RINGS_RX_BUF_LEN, req->rx_buf_len);
+       if (req->_present.tcp_data_split)
+               mnl_attr_put_u8(nlh, ETHTOOL_A_RINGS_TCP_DATA_SPLIT, req->tcp_data_split);
+       if (req->_present.cqe_size)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_RINGS_CQE_SIZE, req->cqe_size);
+       if (req->_present.tx_push)
+               mnl_attr_put_u8(nlh, ETHTOOL_A_RINGS_TX_PUSH, req->tx_push);
+       if (req->_present.rx_push)
+               mnl_attr_put_u8(nlh, ETHTOOL_A_RINGS_RX_PUSH, req->rx_push);
+       if (req->_present.tx_push_buf_len)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_RINGS_TX_PUSH_BUF_LEN, req->tx_push_buf_len);
+       if (req->_present.tx_push_buf_len_max)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_RINGS_TX_PUSH_BUF_LEN_MAX, req->tx_push_buf_len_max);
+
+       err = ynl_exec(ys, nlh, NULL);
+       if (err < 0)
+               return -1;
+
+       return 0;
+}
+
+/* ============== ETHTOOL_MSG_CHANNELS_GET ============== */
+/* ETHTOOL_MSG_CHANNELS_GET - do */
+void ethtool_channels_get_req_free(struct ethtool_channels_get_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+void ethtool_channels_get_rsp_free(struct ethtool_channels_get_rsp *rsp)
+{
+       ethtool_header_free(&rsp->header);
+       free(rsp);
+}
+
+int ethtool_channels_get_rsp_parse(const struct nlmsghdr *nlh, void *data)
+{
+       struct ethtool_channels_get_rsp *dst;
+       struct ynl_parse_arg *yarg = data;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+
+       dst = yarg->data;
+       parg.ys = yarg->ys;
+
+       mnl_attr_for_each(attr, nlh, sizeof(struct genlmsghdr)) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_CHANNELS_HEADER) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.header = 1;
+
+                       parg.rsp_policy = &ethtool_header_nest;
+                       parg.data = &dst->header;
+                       if (ethtool_header_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_CHANNELS_RX_MAX) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.rx_max = 1;
+                       dst->rx_max = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_CHANNELS_TX_MAX) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.tx_max = 1;
+                       dst->tx_max = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_CHANNELS_OTHER_MAX) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.other_max = 1;
+                       dst->other_max = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_CHANNELS_COMBINED_MAX) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.combined_max = 1;
+                       dst->combined_max = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_CHANNELS_RX_COUNT) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.rx_count = 1;
+                       dst->rx_count = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_CHANNELS_TX_COUNT) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.tx_count = 1;
+                       dst->tx_count = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_CHANNELS_OTHER_COUNT) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.other_count = 1;
+                       dst->other_count = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_CHANNELS_COMBINED_COUNT) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.combined_count = 1;
+                       dst->combined_count = mnl_attr_get_u32(attr);
+               }
+       }
+
+       return MNL_CB_OK;
+}
+
+struct ethtool_channels_get_rsp *
+ethtool_channels_get(struct ynl_sock *ys, struct ethtool_channels_get_req *req)
+{
+       struct ynl_req_state yrs = { .yarg = { .ys = ys, }, };
+       struct ethtool_channels_get_rsp *rsp;
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_CHANNELS_GET, 1);
+       ys->req_policy = &ethtool_channels_nest;
+       yrs.yarg.rsp_policy = &ethtool_channels_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_CHANNELS_HEADER, &req->header);
+
+       rsp = calloc(1, sizeof(*rsp));
+       yrs.yarg.data = rsp;
+       yrs.cb = ethtool_channels_get_rsp_parse;
+       yrs.rsp_cmd = 18;
+
+       err = ynl_exec(ys, nlh, &yrs);
+       if (err < 0)
+               goto err_free;
+
+       return rsp;
+
+err_free:
+       ethtool_channels_get_rsp_free(rsp);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_CHANNELS_GET - dump */
+void ethtool_channels_get_list_free(struct ethtool_channels_get_list *rsp)
+{
+       struct ethtool_channels_get_list *next = rsp;
+
+       while ((void *)next != YNL_LIST_END) {
+               rsp = next;
+               next = rsp->next;
+
+               ethtool_header_free(&rsp->obj.header);
+               free(rsp);
+       }
+}
+
+struct ethtool_channels_get_list *
+ethtool_channels_get_dump(struct ynl_sock *ys,
+                         struct ethtool_channels_get_req_dump *req)
+{
+       struct ynl_dump_state yds = {};
+       struct nlmsghdr *nlh;
+       int err;
+
+       yds.ys = ys;
+       yds.alloc_sz = sizeof(struct ethtool_channels_get_list);
+       yds.cb = ethtool_channels_get_rsp_parse;
+       yds.rsp_cmd = 18;
+       yds.rsp_policy = &ethtool_channels_nest;
+
+       nlh = ynl_gemsg_start_dump(ys, ys->family_id, ETHTOOL_MSG_CHANNELS_GET, 1);
+       ys->req_policy = &ethtool_channels_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_CHANNELS_HEADER, &req->header);
+
+       err = ynl_exec_dump(ys, nlh, &yds);
+       if (err < 0)
+               goto free_list;
+
+       return yds.first;
+
+free_list:
+       ethtool_channels_get_list_free(yds.first);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_CHANNELS_GET - notify */
+void ethtool_channels_get_ntf_free(struct ethtool_channels_get_ntf *rsp)
+{
+       ethtool_header_free(&rsp->obj.header);
+       free(rsp);
+}
+
+/* ============== ETHTOOL_MSG_CHANNELS_SET ============== */
+/* ETHTOOL_MSG_CHANNELS_SET - do */
+void ethtool_channels_set_req_free(struct ethtool_channels_set_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+int ethtool_channels_set(struct ynl_sock *ys,
+                        struct ethtool_channels_set_req *req)
+{
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_CHANNELS_SET, 1);
+       ys->req_policy = &ethtool_channels_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_CHANNELS_HEADER, &req->header);
+       if (req->_present.rx_max)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_CHANNELS_RX_MAX, req->rx_max);
+       if (req->_present.tx_max)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_CHANNELS_TX_MAX, req->tx_max);
+       if (req->_present.other_max)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_CHANNELS_OTHER_MAX, req->other_max);
+       if (req->_present.combined_max)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_CHANNELS_COMBINED_MAX, req->combined_max);
+       if (req->_present.rx_count)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_CHANNELS_RX_COUNT, req->rx_count);
+       if (req->_present.tx_count)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_CHANNELS_TX_COUNT, req->tx_count);
+       if (req->_present.other_count)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_CHANNELS_OTHER_COUNT, req->other_count);
+       if (req->_present.combined_count)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_CHANNELS_COMBINED_COUNT, req->combined_count);
+
+       err = ynl_exec(ys, nlh, NULL);
+       if (err < 0)
+               return -1;
+
+       return 0;
+}
+
+/* ============== ETHTOOL_MSG_COALESCE_GET ============== */
+/* ETHTOOL_MSG_COALESCE_GET - do */
+void ethtool_coalesce_get_req_free(struct ethtool_coalesce_get_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+void ethtool_coalesce_get_rsp_free(struct ethtool_coalesce_get_rsp *rsp)
+{
+       ethtool_header_free(&rsp->header);
+       free(rsp);
+}
+
+int ethtool_coalesce_get_rsp_parse(const struct nlmsghdr *nlh, void *data)
+{
+       struct ethtool_coalesce_get_rsp *dst;
+       struct ynl_parse_arg *yarg = data;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+
+       dst = yarg->data;
+       parg.ys = yarg->ys;
+
+       mnl_attr_for_each(attr, nlh, sizeof(struct genlmsghdr)) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_COALESCE_HEADER) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.header = 1;
+
+                       parg.rsp_policy = &ethtool_header_nest;
+                       parg.data = &dst->header;
+                       if (ethtool_header_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_COALESCE_RX_USECS) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.rx_usecs = 1;
+                       dst->rx_usecs = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_COALESCE_RX_MAX_FRAMES) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.rx_max_frames = 1;
+                       dst->rx_max_frames = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_COALESCE_RX_USECS_IRQ) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.rx_usecs_irq = 1;
+                       dst->rx_usecs_irq = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_COALESCE_RX_MAX_FRAMES_IRQ) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.rx_max_frames_irq = 1;
+                       dst->rx_max_frames_irq = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_COALESCE_TX_USECS) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.tx_usecs = 1;
+                       dst->tx_usecs = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_COALESCE_TX_MAX_FRAMES) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.tx_max_frames = 1;
+                       dst->tx_max_frames = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_COALESCE_TX_USECS_IRQ) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.tx_usecs_irq = 1;
+                       dst->tx_usecs_irq = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_COALESCE_TX_MAX_FRAMES_IRQ) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.tx_max_frames_irq = 1;
+                       dst->tx_max_frames_irq = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_COALESCE_STATS_BLOCK_USECS) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.stats_block_usecs = 1;
+                       dst->stats_block_usecs = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_COALESCE_USE_ADAPTIVE_RX) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.use_adaptive_rx = 1;
+                       dst->use_adaptive_rx = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_COALESCE_USE_ADAPTIVE_TX) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.use_adaptive_tx = 1;
+                       dst->use_adaptive_tx = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_COALESCE_PKT_RATE_LOW) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.pkt_rate_low = 1;
+                       dst->pkt_rate_low = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_COALESCE_RX_USECS_LOW) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.rx_usecs_low = 1;
+                       dst->rx_usecs_low = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_COALESCE_RX_MAX_FRAMES_LOW) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.rx_max_frames_low = 1;
+                       dst->rx_max_frames_low = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_COALESCE_TX_USECS_LOW) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.tx_usecs_low = 1;
+                       dst->tx_usecs_low = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_COALESCE_TX_MAX_FRAMES_LOW) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.tx_max_frames_low = 1;
+                       dst->tx_max_frames_low = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_COALESCE_PKT_RATE_HIGH) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.pkt_rate_high = 1;
+                       dst->pkt_rate_high = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_COALESCE_RX_USECS_HIGH) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.rx_usecs_high = 1;
+                       dst->rx_usecs_high = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_COALESCE_RX_MAX_FRAMES_HIGH) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.rx_max_frames_high = 1;
+                       dst->rx_max_frames_high = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_COALESCE_TX_USECS_HIGH) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.tx_usecs_high = 1;
+                       dst->tx_usecs_high = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_COALESCE_TX_MAX_FRAMES_HIGH) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.tx_max_frames_high = 1;
+                       dst->tx_max_frames_high = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_COALESCE_RATE_SAMPLE_INTERVAL) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.rate_sample_interval = 1;
+                       dst->rate_sample_interval = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_COALESCE_USE_CQE_MODE_TX) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.use_cqe_mode_tx = 1;
+                       dst->use_cqe_mode_tx = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_COALESCE_USE_CQE_MODE_RX) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.use_cqe_mode_rx = 1;
+                       dst->use_cqe_mode_rx = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_COALESCE_TX_AGGR_MAX_BYTES) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.tx_aggr_max_bytes = 1;
+                       dst->tx_aggr_max_bytes = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_COALESCE_TX_AGGR_MAX_FRAMES) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.tx_aggr_max_frames = 1;
+                       dst->tx_aggr_max_frames = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_COALESCE_TX_AGGR_TIME_USECS) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.tx_aggr_time_usecs = 1;
+                       dst->tx_aggr_time_usecs = mnl_attr_get_u32(attr);
+               }
+       }
+
+       return MNL_CB_OK;
+}
+
+struct ethtool_coalesce_get_rsp *
+ethtool_coalesce_get(struct ynl_sock *ys, struct ethtool_coalesce_get_req *req)
+{
+       struct ynl_req_state yrs = { .yarg = { .ys = ys, }, };
+       struct ethtool_coalesce_get_rsp *rsp;
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_COALESCE_GET, 1);
+       ys->req_policy = &ethtool_coalesce_nest;
+       yrs.yarg.rsp_policy = &ethtool_coalesce_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_COALESCE_HEADER, &req->header);
+
+       rsp = calloc(1, sizeof(*rsp));
+       yrs.yarg.data = rsp;
+       yrs.cb = ethtool_coalesce_get_rsp_parse;
+       yrs.rsp_cmd = 20;
+
+       err = ynl_exec(ys, nlh, &yrs);
+       if (err < 0)
+               goto err_free;
+
+       return rsp;
+
+err_free:
+       ethtool_coalesce_get_rsp_free(rsp);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_COALESCE_GET - dump */
+void ethtool_coalesce_get_list_free(struct ethtool_coalesce_get_list *rsp)
+{
+       struct ethtool_coalesce_get_list *next = rsp;
+
+       while ((void *)next != YNL_LIST_END) {
+               rsp = next;
+               next = rsp->next;
+
+               ethtool_header_free(&rsp->obj.header);
+               free(rsp);
+       }
+}
+
+struct ethtool_coalesce_get_list *
+ethtool_coalesce_get_dump(struct ynl_sock *ys,
+                         struct ethtool_coalesce_get_req_dump *req)
+{
+       struct ynl_dump_state yds = {};
+       struct nlmsghdr *nlh;
+       int err;
+
+       yds.ys = ys;
+       yds.alloc_sz = sizeof(struct ethtool_coalesce_get_list);
+       yds.cb = ethtool_coalesce_get_rsp_parse;
+       yds.rsp_cmd = 20;
+       yds.rsp_policy = &ethtool_coalesce_nest;
+
+       nlh = ynl_gemsg_start_dump(ys, ys->family_id, ETHTOOL_MSG_COALESCE_GET, 1);
+       ys->req_policy = &ethtool_coalesce_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_COALESCE_HEADER, &req->header);
+
+       err = ynl_exec_dump(ys, nlh, &yds);
+       if (err < 0)
+               goto free_list;
+
+       return yds.first;
+
+free_list:
+       ethtool_coalesce_get_list_free(yds.first);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_COALESCE_GET - notify */
+void ethtool_coalesce_get_ntf_free(struct ethtool_coalesce_get_ntf *rsp)
+{
+       ethtool_header_free(&rsp->obj.header);
+       free(rsp);
+}
+
+/* ============== ETHTOOL_MSG_COALESCE_SET ============== */
+/* ETHTOOL_MSG_COALESCE_SET - do */
+void ethtool_coalesce_set_req_free(struct ethtool_coalesce_set_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+int ethtool_coalesce_set(struct ynl_sock *ys,
+                        struct ethtool_coalesce_set_req *req)
+{
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_COALESCE_SET, 1);
+       ys->req_policy = &ethtool_coalesce_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_COALESCE_HEADER, &req->header);
+       if (req->_present.rx_usecs)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_COALESCE_RX_USECS, req->rx_usecs);
+       if (req->_present.rx_max_frames)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_COALESCE_RX_MAX_FRAMES, req->rx_max_frames);
+       if (req->_present.rx_usecs_irq)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_COALESCE_RX_USECS_IRQ, req->rx_usecs_irq);
+       if (req->_present.rx_max_frames_irq)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_COALESCE_RX_MAX_FRAMES_IRQ, req->rx_max_frames_irq);
+       if (req->_present.tx_usecs)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_COALESCE_TX_USECS, req->tx_usecs);
+       if (req->_present.tx_max_frames)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_COALESCE_TX_MAX_FRAMES, req->tx_max_frames);
+       if (req->_present.tx_usecs_irq)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_COALESCE_TX_USECS_IRQ, req->tx_usecs_irq);
+       if (req->_present.tx_max_frames_irq)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_COALESCE_TX_MAX_FRAMES_IRQ, req->tx_max_frames_irq);
+       if (req->_present.stats_block_usecs)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_COALESCE_STATS_BLOCK_USECS, req->stats_block_usecs);
+       if (req->_present.use_adaptive_rx)
+               mnl_attr_put_u8(nlh, ETHTOOL_A_COALESCE_USE_ADAPTIVE_RX, req->use_adaptive_rx);
+       if (req->_present.use_adaptive_tx)
+               mnl_attr_put_u8(nlh, ETHTOOL_A_COALESCE_USE_ADAPTIVE_TX, req->use_adaptive_tx);
+       if (req->_present.pkt_rate_low)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_COALESCE_PKT_RATE_LOW, req->pkt_rate_low);
+       if (req->_present.rx_usecs_low)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_COALESCE_RX_USECS_LOW, req->rx_usecs_low);
+       if (req->_present.rx_max_frames_low)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_COALESCE_RX_MAX_FRAMES_LOW, req->rx_max_frames_low);
+       if (req->_present.tx_usecs_low)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_COALESCE_TX_USECS_LOW, req->tx_usecs_low);
+       if (req->_present.tx_max_frames_low)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_COALESCE_TX_MAX_FRAMES_LOW, req->tx_max_frames_low);
+       if (req->_present.pkt_rate_high)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_COALESCE_PKT_RATE_HIGH, req->pkt_rate_high);
+       if (req->_present.rx_usecs_high)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_COALESCE_RX_USECS_HIGH, req->rx_usecs_high);
+       if (req->_present.rx_max_frames_high)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_COALESCE_RX_MAX_FRAMES_HIGH, req->rx_max_frames_high);
+       if (req->_present.tx_usecs_high)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_COALESCE_TX_USECS_HIGH, req->tx_usecs_high);
+       if (req->_present.tx_max_frames_high)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_COALESCE_TX_MAX_FRAMES_HIGH, req->tx_max_frames_high);
+       if (req->_present.rate_sample_interval)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_COALESCE_RATE_SAMPLE_INTERVAL, req->rate_sample_interval);
+       if (req->_present.use_cqe_mode_tx)
+               mnl_attr_put_u8(nlh, ETHTOOL_A_COALESCE_USE_CQE_MODE_TX, req->use_cqe_mode_tx);
+       if (req->_present.use_cqe_mode_rx)
+               mnl_attr_put_u8(nlh, ETHTOOL_A_COALESCE_USE_CQE_MODE_RX, req->use_cqe_mode_rx);
+       if (req->_present.tx_aggr_max_bytes)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_COALESCE_TX_AGGR_MAX_BYTES, req->tx_aggr_max_bytes);
+       if (req->_present.tx_aggr_max_frames)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_COALESCE_TX_AGGR_MAX_FRAMES, req->tx_aggr_max_frames);
+       if (req->_present.tx_aggr_time_usecs)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_COALESCE_TX_AGGR_TIME_USECS, req->tx_aggr_time_usecs);
+
+       err = ynl_exec(ys, nlh, NULL);
+       if (err < 0)
+               return -1;
+
+       return 0;
+}
+
+/* ============== ETHTOOL_MSG_PAUSE_GET ============== */
+/* ETHTOOL_MSG_PAUSE_GET - do */
+void ethtool_pause_get_req_free(struct ethtool_pause_get_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+void ethtool_pause_get_rsp_free(struct ethtool_pause_get_rsp *rsp)
+{
+       ethtool_header_free(&rsp->header);
+       ethtool_pause_stat_free(&rsp->stats);
+       free(rsp);
+}
+
+int ethtool_pause_get_rsp_parse(const struct nlmsghdr *nlh, void *data)
+{
+       struct ethtool_pause_get_rsp *dst;
+       struct ynl_parse_arg *yarg = data;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+
+       dst = yarg->data;
+       parg.ys = yarg->ys;
+
+       mnl_attr_for_each(attr, nlh, sizeof(struct genlmsghdr)) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_PAUSE_HEADER) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.header = 1;
+
+                       parg.rsp_policy = &ethtool_header_nest;
+                       parg.data = &dst->header;
+                       if (ethtool_header_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_PAUSE_AUTONEG) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.autoneg = 1;
+                       dst->autoneg = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_PAUSE_RX) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.rx = 1;
+                       dst->rx = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_PAUSE_TX) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.tx = 1;
+                       dst->tx = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_PAUSE_STATS) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.stats = 1;
+
+                       parg.rsp_policy = &ethtool_pause_stat_nest;
+                       parg.data = &dst->stats;
+                       if (ethtool_pause_stat_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_PAUSE_STATS_SRC) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.stats_src = 1;
+                       dst->stats_src = mnl_attr_get_u32(attr);
+               }
+       }
+
+       return MNL_CB_OK;
+}
+
+struct ethtool_pause_get_rsp *
+ethtool_pause_get(struct ynl_sock *ys, struct ethtool_pause_get_req *req)
+{
+       struct ynl_req_state yrs = { .yarg = { .ys = ys, }, };
+       struct ethtool_pause_get_rsp *rsp;
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_PAUSE_GET, 1);
+       ys->req_policy = &ethtool_pause_nest;
+       yrs.yarg.rsp_policy = &ethtool_pause_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_PAUSE_HEADER, &req->header);
+
+       rsp = calloc(1, sizeof(*rsp));
+       yrs.yarg.data = rsp;
+       yrs.cb = ethtool_pause_get_rsp_parse;
+       yrs.rsp_cmd = 22;
+
+       err = ynl_exec(ys, nlh, &yrs);
+       if (err < 0)
+               goto err_free;
+
+       return rsp;
+
+err_free:
+       ethtool_pause_get_rsp_free(rsp);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_PAUSE_GET - dump */
+void ethtool_pause_get_list_free(struct ethtool_pause_get_list *rsp)
+{
+       struct ethtool_pause_get_list *next = rsp;
+
+       while ((void *)next != YNL_LIST_END) {
+               rsp = next;
+               next = rsp->next;
+
+               ethtool_header_free(&rsp->obj.header);
+               ethtool_pause_stat_free(&rsp->obj.stats);
+               free(rsp);
+       }
+}
+
+struct ethtool_pause_get_list *
+ethtool_pause_get_dump(struct ynl_sock *ys,
+                      struct ethtool_pause_get_req_dump *req)
+{
+       struct ynl_dump_state yds = {};
+       struct nlmsghdr *nlh;
+       int err;
+
+       yds.ys = ys;
+       yds.alloc_sz = sizeof(struct ethtool_pause_get_list);
+       yds.cb = ethtool_pause_get_rsp_parse;
+       yds.rsp_cmd = 22;
+       yds.rsp_policy = &ethtool_pause_nest;
+
+       nlh = ynl_gemsg_start_dump(ys, ys->family_id, ETHTOOL_MSG_PAUSE_GET, 1);
+       ys->req_policy = &ethtool_pause_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_PAUSE_HEADER, &req->header);
+
+       err = ynl_exec_dump(ys, nlh, &yds);
+       if (err < 0)
+               goto free_list;
+
+       return yds.first;
+
+free_list:
+       ethtool_pause_get_list_free(yds.first);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_PAUSE_GET - notify */
+void ethtool_pause_get_ntf_free(struct ethtool_pause_get_ntf *rsp)
+{
+       ethtool_header_free(&rsp->obj.header);
+       ethtool_pause_stat_free(&rsp->obj.stats);
+       free(rsp);
+}
+
+/* ============== ETHTOOL_MSG_PAUSE_SET ============== */
+/* ETHTOOL_MSG_PAUSE_SET - do */
+void ethtool_pause_set_req_free(struct ethtool_pause_set_req *req)
+{
+       ethtool_header_free(&req->header);
+       ethtool_pause_stat_free(&req->stats);
+       free(req);
+}
+
+int ethtool_pause_set(struct ynl_sock *ys, struct ethtool_pause_set_req *req)
+{
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_PAUSE_SET, 1);
+       ys->req_policy = &ethtool_pause_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_PAUSE_HEADER, &req->header);
+       if (req->_present.autoneg)
+               mnl_attr_put_u8(nlh, ETHTOOL_A_PAUSE_AUTONEG, req->autoneg);
+       if (req->_present.rx)
+               mnl_attr_put_u8(nlh, ETHTOOL_A_PAUSE_RX, req->rx);
+       if (req->_present.tx)
+               mnl_attr_put_u8(nlh, ETHTOOL_A_PAUSE_TX, req->tx);
+       if (req->_present.stats)
+               ethtool_pause_stat_put(nlh, ETHTOOL_A_PAUSE_STATS, &req->stats);
+       if (req->_present.stats_src)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_PAUSE_STATS_SRC, req->stats_src);
+
+       err = ynl_exec(ys, nlh, NULL);
+       if (err < 0)
+               return -1;
+
+       return 0;
+}
+
+/* ============== ETHTOOL_MSG_EEE_GET ============== */
+/* ETHTOOL_MSG_EEE_GET - do */
+void ethtool_eee_get_req_free(struct ethtool_eee_get_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+void ethtool_eee_get_rsp_free(struct ethtool_eee_get_rsp *rsp)
+{
+       ethtool_header_free(&rsp->header);
+       ethtool_bitset_free(&rsp->modes_ours);
+       ethtool_bitset_free(&rsp->modes_peer);
+       free(rsp);
+}
+
+int ethtool_eee_get_rsp_parse(const struct nlmsghdr *nlh, void *data)
+{
+       struct ynl_parse_arg *yarg = data;
+       struct ethtool_eee_get_rsp *dst;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+
+       dst = yarg->data;
+       parg.ys = yarg->ys;
+
+       mnl_attr_for_each(attr, nlh, sizeof(struct genlmsghdr)) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_EEE_HEADER) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.header = 1;
+
+                       parg.rsp_policy = &ethtool_header_nest;
+                       parg.data = &dst->header;
+                       if (ethtool_header_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_EEE_MODES_OURS) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.modes_ours = 1;
+
+                       parg.rsp_policy = &ethtool_bitset_nest;
+                       parg.data = &dst->modes_ours;
+                       if (ethtool_bitset_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_EEE_MODES_PEER) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.modes_peer = 1;
+
+                       parg.rsp_policy = &ethtool_bitset_nest;
+                       parg.data = &dst->modes_peer;
+                       if (ethtool_bitset_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_EEE_ACTIVE) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.active = 1;
+                       dst->active = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_EEE_ENABLED) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.enabled = 1;
+                       dst->enabled = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_EEE_TX_LPI_ENABLED) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.tx_lpi_enabled = 1;
+                       dst->tx_lpi_enabled = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_EEE_TX_LPI_TIMER) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.tx_lpi_timer = 1;
+                       dst->tx_lpi_timer = mnl_attr_get_u32(attr);
+               }
+       }
+
+       return MNL_CB_OK;
+}
+
+struct ethtool_eee_get_rsp *
+ethtool_eee_get(struct ynl_sock *ys, struct ethtool_eee_get_req *req)
+{
+       struct ynl_req_state yrs = { .yarg = { .ys = ys, }, };
+       struct ethtool_eee_get_rsp *rsp;
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_EEE_GET, 1);
+       ys->req_policy = &ethtool_eee_nest;
+       yrs.yarg.rsp_policy = &ethtool_eee_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_EEE_HEADER, &req->header);
+
+       rsp = calloc(1, sizeof(*rsp));
+       yrs.yarg.data = rsp;
+       yrs.cb = ethtool_eee_get_rsp_parse;
+       yrs.rsp_cmd = 24;
+
+       err = ynl_exec(ys, nlh, &yrs);
+       if (err < 0)
+               goto err_free;
+
+       return rsp;
+
+err_free:
+       ethtool_eee_get_rsp_free(rsp);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_EEE_GET - dump */
+void ethtool_eee_get_list_free(struct ethtool_eee_get_list *rsp)
+{
+       struct ethtool_eee_get_list *next = rsp;
+
+       while ((void *)next != YNL_LIST_END) {
+               rsp = next;
+               next = rsp->next;
+
+               ethtool_header_free(&rsp->obj.header);
+               ethtool_bitset_free(&rsp->obj.modes_ours);
+               ethtool_bitset_free(&rsp->obj.modes_peer);
+               free(rsp);
+       }
+}
+
+struct ethtool_eee_get_list *
+ethtool_eee_get_dump(struct ynl_sock *ys, struct ethtool_eee_get_req_dump *req)
+{
+       struct ynl_dump_state yds = {};
+       struct nlmsghdr *nlh;
+       int err;
+
+       yds.ys = ys;
+       yds.alloc_sz = sizeof(struct ethtool_eee_get_list);
+       yds.cb = ethtool_eee_get_rsp_parse;
+       yds.rsp_cmd = 24;
+       yds.rsp_policy = &ethtool_eee_nest;
+
+       nlh = ynl_gemsg_start_dump(ys, ys->family_id, ETHTOOL_MSG_EEE_GET, 1);
+       ys->req_policy = &ethtool_eee_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_EEE_HEADER, &req->header);
+
+       err = ynl_exec_dump(ys, nlh, &yds);
+       if (err < 0)
+               goto free_list;
+
+       return yds.first;
+
+free_list:
+       ethtool_eee_get_list_free(yds.first);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_EEE_GET - notify */
+void ethtool_eee_get_ntf_free(struct ethtool_eee_get_ntf *rsp)
+{
+       ethtool_header_free(&rsp->obj.header);
+       ethtool_bitset_free(&rsp->obj.modes_ours);
+       ethtool_bitset_free(&rsp->obj.modes_peer);
+       free(rsp);
+}
+
+/* ============== ETHTOOL_MSG_EEE_SET ============== */
+/* ETHTOOL_MSG_EEE_SET - do */
+void ethtool_eee_set_req_free(struct ethtool_eee_set_req *req)
+{
+       ethtool_header_free(&req->header);
+       ethtool_bitset_free(&req->modes_ours);
+       ethtool_bitset_free(&req->modes_peer);
+       free(req);
+}
+
+int ethtool_eee_set(struct ynl_sock *ys, struct ethtool_eee_set_req *req)
+{
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_EEE_SET, 1);
+       ys->req_policy = &ethtool_eee_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_EEE_HEADER, &req->header);
+       if (req->_present.modes_ours)
+               ethtool_bitset_put(nlh, ETHTOOL_A_EEE_MODES_OURS, &req->modes_ours);
+       if (req->_present.modes_peer)
+               ethtool_bitset_put(nlh, ETHTOOL_A_EEE_MODES_PEER, &req->modes_peer);
+       if (req->_present.active)
+               mnl_attr_put_u8(nlh, ETHTOOL_A_EEE_ACTIVE, req->active);
+       if (req->_present.enabled)
+               mnl_attr_put_u8(nlh, ETHTOOL_A_EEE_ENABLED, req->enabled);
+       if (req->_present.tx_lpi_enabled)
+               mnl_attr_put_u8(nlh, ETHTOOL_A_EEE_TX_LPI_ENABLED, req->tx_lpi_enabled);
+       if (req->_present.tx_lpi_timer)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_EEE_TX_LPI_TIMER, req->tx_lpi_timer);
+
+       err = ynl_exec(ys, nlh, NULL);
+       if (err < 0)
+               return -1;
+
+       return 0;
+}
+
+/* ============== ETHTOOL_MSG_TSINFO_GET ============== */
+/* ETHTOOL_MSG_TSINFO_GET - do */
+void ethtool_tsinfo_get_req_free(struct ethtool_tsinfo_get_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+void ethtool_tsinfo_get_rsp_free(struct ethtool_tsinfo_get_rsp *rsp)
+{
+       ethtool_header_free(&rsp->header);
+       ethtool_bitset_free(&rsp->timestamping);
+       ethtool_bitset_free(&rsp->tx_types);
+       ethtool_bitset_free(&rsp->rx_filters);
+       free(rsp);
+}
+
+int ethtool_tsinfo_get_rsp_parse(const struct nlmsghdr *nlh, void *data)
+{
+       struct ethtool_tsinfo_get_rsp *dst;
+       struct ynl_parse_arg *yarg = data;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+
+       dst = yarg->data;
+       parg.ys = yarg->ys;
+
+       mnl_attr_for_each(attr, nlh, sizeof(struct genlmsghdr)) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_TSINFO_HEADER) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.header = 1;
+
+                       parg.rsp_policy = &ethtool_header_nest;
+                       parg.data = &dst->header;
+                       if (ethtool_header_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_TSINFO_TIMESTAMPING) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.timestamping = 1;
+
+                       parg.rsp_policy = &ethtool_bitset_nest;
+                       parg.data = &dst->timestamping;
+                       if (ethtool_bitset_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_TSINFO_TX_TYPES) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.tx_types = 1;
+
+                       parg.rsp_policy = &ethtool_bitset_nest;
+                       parg.data = &dst->tx_types;
+                       if (ethtool_bitset_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_TSINFO_RX_FILTERS) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.rx_filters = 1;
+
+                       parg.rsp_policy = &ethtool_bitset_nest;
+                       parg.data = &dst->rx_filters;
+                       if (ethtool_bitset_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_TSINFO_PHC_INDEX) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.phc_index = 1;
+                       dst->phc_index = mnl_attr_get_u32(attr);
+               }
+       }
+
+       return MNL_CB_OK;
+}
+
+struct ethtool_tsinfo_get_rsp *
+ethtool_tsinfo_get(struct ynl_sock *ys, struct ethtool_tsinfo_get_req *req)
+{
+       struct ynl_req_state yrs = { .yarg = { .ys = ys, }, };
+       struct ethtool_tsinfo_get_rsp *rsp;
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_TSINFO_GET, 1);
+       ys->req_policy = &ethtool_tsinfo_nest;
+       yrs.yarg.rsp_policy = &ethtool_tsinfo_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_TSINFO_HEADER, &req->header);
+
+       rsp = calloc(1, sizeof(*rsp));
+       yrs.yarg.data = rsp;
+       yrs.cb = ethtool_tsinfo_get_rsp_parse;
+       yrs.rsp_cmd = 26;
+
+       err = ynl_exec(ys, nlh, &yrs);
+       if (err < 0)
+               goto err_free;
+
+       return rsp;
+
+err_free:
+       ethtool_tsinfo_get_rsp_free(rsp);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_TSINFO_GET - dump */
+void ethtool_tsinfo_get_list_free(struct ethtool_tsinfo_get_list *rsp)
+{
+       struct ethtool_tsinfo_get_list *next = rsp;
+
+       while ((void *)next != YNL_LIST_END) {
+               rsp = next;
+               next = rsp->next;
+
+               ethtool_header_free(&rsp->obj.header);
+               ethtool_bitset_free(&rsp->obj.timestamping);
+               ethtool_bitset_free(&rsp->obj.tx_types);
+               ethtool_bitset_free(&rsp->obj.rx_filters);
+               free(rsp);
+       }
+}
+
+struct ethtool_tsinfo_get_list *
+ethtool_tsinfo_get_dump(struct ynl_sock *ys,
+                       struct ethtool_tsinfo_get_req_dump *req)
+{
+       struct ynl_dump_state yds = {};
+       struct nlmsghdr *nlh;
+       int err;
+
+       yds.ys = ys;
+       yds.alloc_sz = sizeof(struct ethtool_tsinfo_get_list);
+       yds.cb = ethtool_tsinfo_get_rsp_parse;
+       yds.rsp_cmd = 26;
+       yds.rsp_policy = &ethtool_tsinfo_nest;
+
+       nlh = ynl_gemsg_start_dump(ys, ys->family_id, ETHTOOL_MSG_TSINFO_GET, 1);
+       ys->req_policy = &ethtool_tsinfo_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_TSINFO_HEADER, &req->header);
+
+       err = ynl_exec_dump(ys, nlh, &yds);
+       if (err < 0)
+               goto free_list;
+
+       return yds.first;
+
+free_list:
+       ethtool_tsinfo_get_list_free(yds.first);
+       return NULL;
+}
+
+/* ============== ETHTOOL_MSG_CABLE_TEST_ACT ============== */
+/* ETHTOOL_MSG_CABLE_TEST_ACT - do */
+void ethtool_cable_test_act_req_free(struct ethtool_cable_test_act_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+int ethtool_cable_test_act(struct ynl_sock *ys,
+                          struct ethtool_cable_test_act_req *req)
+{
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_CABLE_TEST_ACT, 1);
+       ys->req_policy = &ethtool_cable_test_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_CABLE_TEST_HEADER, &req->header);
+
+       err = ynl_exec(ys, nlh, NULL);
+       if (err < 0)
+               return -1;
+
+       return 0;
+}
+
+/* ============== ETHTOOL_MSG_CABLE_TEST_TDR_ACT ============== */
+/* ETHTOOL_MSG_CABLE_TEST_TDR_ACT - do */
+void
+ethtool_cable_test_tdr_act_req_free(struct ethtool_cable_test_tdr_act_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+int ethtool_cable_test_tdr_act(struct ynl_sock *ys,
+                              struct ethtool_cable_test_tdr_act_req *req)
+{
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_CABLE_TEST_TDR_ACT, 1);
+       ys->req_policy = &ethtool_cable_test_tdr_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_CABLE_TEST_TDR_HEADER, &req->header);
+
+       err = ynl_exec(ys, nlh, NULL);
+       if (err < 0)
+               return -1;
+
+       return 0;
+}
+
+/* ============== ETHTOOL_MSG_TUNNEL_INFO_GET ============== */
+/* ETHTOOL_MSG_TUNNEL_INFO_GET - do */
+void ethtool_tunnel_info_get_req_free(struct ethtool_tunnel_info_get_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+void ethtool_tunnel_info_get_rsp_free(struct ethtool_tunnel_info_get_rsp *rsp)
+{
+       ethtool_header_free(&rsp->header);
+       ethtool_tunnel_udp_free(&rsp->udp_ports);
+       free(rsp);
+}
+
+int ethtool_tunnel_info_get_rsp_parse(const struct nlmsghdr *nlh, void *data)
+{
+       struct ethtool_tunnel_info_get_rsp *dst;
+       struct ynl_parse_arg *yarg = data;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+
+       dst = yarg->data;
+       parg.ys = yarg->ys;
+
+       mnl_attr_for_each(attr, nlh, sizeof(struct genlmsghdr)) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_TUNNEL_INFO_HEADER) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.header = 1;
+
+                       parg.rsp_policy = &ethtool_header_nest;
+                       parg.data = &dst->header;
+                       if (ethtool_header_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_TUNNEL_INFO_UDP_PORTS) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.udp_ports = 1;
+
+                       parg.rsp_policy = &ethtool_tunnel_udp_nest;
+                       parg.data = &dst->udp_ports;
+                       if (ethtool_tunnel_udp_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               }
+       }
+
+       return MNL_CB_OK;
+}
+
+struct ethtool_tunnel_info_get_rsp *
+ethtool_tunnel_info_get(struct ynl_sock *ys,
+                       struct ethtool_tunnel_info_get_req *req)
+{
+       struct ynl_req_state yrs = { .yarg = { .ys = ys, }, };
+       struct ethtool_tunnel_info_get_rsp *rsp;
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_TUNNEL_INFO_GET, 1);
+       ys->req_policy = &ethtool_tunnel_info_nest;
+       yrs.yarg.rsp_policy = &ethtool_tunnel_info_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_TUNNEL_INFO_HEADER, &req->header);
+
+       rsp = calloc(1, sizeof(*rsp));
+       yrs.yarg.data = rsp;
+       yrs.cb = ethtool_tunnel_info_get_rsp_parse;
+       yrs.rsp_cmd = 29;
+
+       err = ynl_exec(ys, nlh, &yrs);
+       if (err < 0)
+               goto err_free;
+
+       return rsp;
+
+err_free:
+       ethtool_tunnel_info_get_rsp_free(rsp);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_TUNNEL_INFO_GET - dump */
+void
+ethtool_tunnel_info_get_list_free(struct ethtool_tunnel_info_get_list *rsp)
+{
+       struct ethtool_tunnel_info_get_list *next = rsp;
+
+       while ((void *)next != YNL_LIST_END) {
+               rsp = next;
+               next = rsp->next;
+
+               ethtool_header_free(&rsp->obj.header);
+               ethtool_tunnel_udp_free(&rsp->obj.udp_ports);
+               free(rsp);
+       }
+}
+
+struct ethtool_tunnel_info_get_list *
+ethtool_tunnel_info_get_dump(struct ynl_sock *ys,
+                            struct ethtool_tunnel_info_get_req_dump *req)
+{
+       struct ynl_dump_state yds = {};
+       struct nlmsghdr *nlh;
+       int err;
+
+       yds.ys = ys;
+       yds.alloc_sz = sizeof(struct ethtool_tunnel_info_get_list);
+       yds.cb = ethtool_tunnel_info_get_rsp_parse;
+       yds.rsp_cmd = 29;
+       yds.rsp_policy = &ethtool_tunnel_info_nest;
+
+       nlh = ynl_gemsg_start_dump(ys, ys->family_id, ETHTOOL_MSG_TUNNEL_INFO_GET, 1);
+       ys->req_policy = &ethtool_tunnel_info_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_TUNNEL_INFO_HEADER, &req->header);
+
+       err = ynl_exec_dump(ys, nlh, &yds);
+       if (err < 0)
+               goto free_list;
+
+       return yds.first;
+
+free_list:
+       ethtool_tunnel_info_get_list_free(yds.first);
+       return NULL;
+}
+
+/* ============== ETHTOOL_MSG_FEC_GET ============== */
+/* ETHTOOL_MSG_FEC_GET - do */
+void ethtool_fec_get_req_free(struct ethtool_fec_get_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+void ethtool_fec_get_rsp_free(struct ethtool_fec_get_rsp *rsp)
+{
+       ethtool_header_free(&rsp->header);
+       ethtool_bitset_free(&rsp->modes);
+       ethtool_fec_stat_free(&rsp->stats);
+       free(rsp);
+}
+
+int ethtool_fec_get_rsp_parse(const struct nlmsghdr *nlh, void *data)
+{
+       struct ynl_parse_arg *yarg = data;
+       struct ethtool_fec_get_rsp *dst;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+
+       dst = yarg->data;
+       parg.ys = yarg->ys;
+
+       mnl_attr_for_each(attr, nlh, sizeof(struct genlmsghdr)) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_FEC_HEADER) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.header = 1;
+
+                       parg.rsp_policy = &ethtool_header_nest;
+                       parg.data = &dst->header;
+                       if (ethtool_header_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_FEC_MODES) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.modes = 1;
+
+                       parg.rsp_policy = &ethtool_bitset_nest;
+                       parg.data = &dst->modes;
+                       if (ethtool_bitset_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_FEC_AUTO) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.auto_ = 1;
+                       dst->auto_ = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_FEC_ACTIVE) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.active = 1;
+                       dst->active = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_FEC_STATS) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.stats = 1;
+
+                       parg.rsp_policy = &ethtool_fec_stat_nest;
+                       parg.data = &dst->stats;
+                       if (ethtool_fec_stat_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               }
+       }
+
+       return MNL_CB_OK;
+}
+
+struct ethtool_fec_get_rsp *
+ethtool_fec_get(struct ynl_sock *ys, struct ethtool_fec_get_req *req)
+{
+       struct ynl_req_state yrs = { .yarg = { .ys = ys, }, };
+       struct ethtool_fec_get_rsp *rsp;
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_FEC_GET, 1);
+       ys->req_policy = &ethtool_fec_nest;
+       yrs.yarg.rsp_policy = &ethtool_fec_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_FEC_HEADER, &req->header);
+
+       rsp = calloc(1, sizeof(*rsp));
+       yrs.yarg.data = rsp;
+       yrs.cb = ethtool_fec_get_rsp_parse;
+       yrs.rsp_cmd = 30;
+
+       err = ynl_exec(ys, nlh, &yrs);
+       if (err < 0)
+               goto err_free;
+
+       return rsp;
+
+err_free:
+       ethtool_fec_get_rsp_free(rsp);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_FEC_GET - dump */
+void ethtool_fec_get_list_free(struct ethtool_fec_get_list *rsp)
+{
+       struct ethtool_fec_get_list *next = rsp;
+
+       while ((void *)next != YNL_LIST_END) {
+               rsp = next;
+               next = rsp->next;
+
+               ethtool_header_free(&rsp->obj.header);
+               ethtool_bitset_free(&rsp->obj.modes);
+               ethtool_fec_stat_free(&rsp->obj.stats);
+               free(rsp);
+       }
+}
+
+struct ethtool_fec_get_list *
+ethtool_fec_get_dump(struct ynl_sock *ys, struct ethtool_fec_get_req_dump *req)
+{
+       struct ynl_dump_state yds = {};
+       struct nlmsghdr *nlh;
+       int err;
+
+       yds.ys = ys;
+       yds.alloc_sz = sizeof(struct ethtool_fec_get_list);
+       yds.cb = ethtool_fec_get_rsp_parse;
+       yds.rsp_cmd = 30;
+       yds.rsp_policy = &ethtool_fec_nest;
+
+       nlh = ynl_gemsg_start_dump(ys, ys->family_id, ETHTOOL_MSG_FEC_GET, 1);
+       ys->req_policy = &ethtool_fec_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_FEC_HEADER, &req->header);
+
+       err = ynl_exec_dump(ys, nlh, &yds);
+       if (err < 0)
+               goto free_list;
+
+       return yds.first;
+
+free_list:
+       ethtool_fec_get_list_free(yds.first);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_FEC_GET - notify */
+void ethtool_fec_get_ntf_free(struct ethtool_fec_get_ntf *rsp)
+{
+       ethtool_header_free(&rsp->obj.header);
+       ethtool_bitset_free(&rsp->obj.modes);
+       ethtool_fec_stat_free(&rsp->obj.stats);
+       free(rsp);
+}
+
+/* ============== ETHTOOL_MSG_FEC_SET ============== */
+/* ETHTOOL_MSG_FEC_SET - do */
+void ethtool_fec_set_req_free(struct ethtool_fec_set_req *req)
+{
+       ethtool_header_free(&req->header);
+       ethtool_bitset_free(&req->modes);
+       ethtool_fec_stat_free(&req->stats);
+       free(req);
+}
+
+int ethtool_fec_set(struct ynl_sock *ys, struct ethtool_fec_set_req *req)
+{
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_FEC_SET, 1);
+       ys->req_policy = &ethtool_fec_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_FEC_HEADER, &req->header);
+       if (req->_present.modes)
+               ethtool_bitset_put(nlh, ETHTOOL_A_FEC_MODES, &req->modes);
+       if (req->_present.auto_)
+               mnl_attr_put_u8(nlh, ETHTOOL_A_FEC_AUTO, req->auto_);
+       if (req->_present.active)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_FEC_ACTIVE, req->active);
+       if (req->_present.stats)
+               ethtool_fec_stat_put(nlh, ETHTOOL_A_FEC_STATS, &req->stats);
+
+       err = ynl_exec(ys, nlh, NULL);
+       if (err < 0)
+               return -1;
+
+       return 0;
+}
+
+/* ============== ETHTOOL_MSG_MODULE_EEPROM_GET ============== */
+/* ETHTOOL_MSG_MODULE_EEPROM_GET - do */
+void
+ethtool_module_eeprom_get_req_free(struct ethtool_module_eeprom_get_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+void
+ethtool_module_eeprom_get_rsp_free(struct ethtool_module_eeprom_get_rsp *rsp)
+{
+       ethtool_header_free(&rsp->header);
+       free(rsp->data);
+       free(rsp);
+}
+
+int ethtool_module_eeprom_get_rsp_parse(const struct nlmsghdr *nlh, void *data)
+{
+       struct ethtool_module_eeprom_get_rsp *dst;
+       struct ynl_parse_arg *yarg = data;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+
+       dst = yarg->data;
+       parg.ys = yarg->ys;
+
+       mnl_attr_for_each(attr, nlh, sizeof(struct genlmsghdr)) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_MODULE_EEPROM_HEADER) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.header = 1;
+
+                       parg.rsp_policy = &ethtool_header_nest;
+                       parg.data = &dst->header;
+                       if (ethtool_header_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_MODULE_EEPROM_OFFSET) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.offset = 1;
+                       dst->offset = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_MODULE_EEPROM_LENGTH) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.length = 1;
+                       dst->length = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_MODULE_EEPROM_PAGE) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.page = 1;
+                       dst->page = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_MODULE_EEPROM_BANK) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.bank = 1;
+                       dst->bank = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_MODULE_EEPROM_I2C_ADDRESS) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.i2c_address = 1;
+                       dst->i2c_address = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_MODULE_EEPROM_DATA) {
+                       unsigned int len;
+
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+
+                       len = mnl_attr_get_payload_len(attr);
+                       dst->_present.data_len = len;
+                       dst->data = malloc(len);
+                       memcpy(dst->data, mnl_attr_get_payload(attr), len);
+               }
+       }
+
+       return MNL_CB_OK;
+}
+
+struct ethtool_module_eeprom_get_rsp *
+ethtool_module_eeprom_get(struct ynl_sock *ys,
+                         struct ethtool_module_eeprom_get_req *req)
+{
+       struct ynl_req_state yrs = { .yarg = { .ys = ys, }, };
+       struct ethtool_module_eeprom_get_rsp *rsp;
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_MODULE_EEPROM_GET, 1);
+       ys->req_policy = &ethtool_module_eeprom_nest;
+       yrs.yarg.rsp_policy = &ethtool_module_eeprom_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_MODULE_EEPROM_HEADER, &req->header);
+
+       rsp = calloc(1, sizeof(*rsp));
+       yrs.yarg.data = rsp;
+       yrs.cb = ethtool_module_eeprom_get_rsp_parse;
+       yrs.rsp_cmd = 32;
+
+       err = ynl_exec(ys, nlh, &yrs);
+       if (err < 0)
+               goto err_free;
+
+       return rsp;
+
+err_free:
+       ethtool_module_eeprom_get_rsp_free(rsp);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_MODULE_EEPROM_GET - dump */
+void
+ethtool_module_eeprom_get_list_free(struct ethtool_module_eeprom_get_list *rsp)
+{
+       struct ethtool_module_eeprom_get_list *next = rsp;
+
+       while ((void *)next != YNL_LIST_END) {
+               rsp = next;
+               next = rsp->next;
+
+               ethtool_header_free(&rsp->obj.header);
+               free(rsp->obj.data);
+               free(rsp);
+       }
+}
+
+struct ethtool_module_eeprom_get_list *
+ethtool_module_eeprom_get_dump(struct ynl_sock *ys,
+                              struct ethtool_module_eeprom_get_req_dump *req)
+{
+       struct ynl_dump_state yds = {};
+       struct nlmsghdr *nlh;
+       int err;
+
+       yds.ys = ys;
+       yds.alloc_sz = sizeof(struct ethtool_module_eeprom_get_list);
+       yds.cb = ethtool_module_eeprom_get_rsp_parse;
+       yds.rsp_cmd = 32;
+       yds.rsp_policy = &ethtool_module_eeprom_nest;
+
+       nlh = ynl_gemsg_start_dump(ys, ys->family_id, ETHTOOL_MSG_MODULE_EEPROM_GET, 1);
+       ys->req_policy = &ethtool_module_eeprom_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_MODULE_EEPROM_HEADER, &req->header);
+
+       err = ynl_exec_dump(ys, nlh, &yds);
+       if (err < 0)
+               goto free_list;
+
+       return yds.first;
+
+free_list:
+       ethtool_module_eeprom_get_list_free(yds.first);
+       return NULL;
+}
+
+/* ============== ETHTOOL_MSG_PHC_VCLOCKS_GET ============== */
+/* ETHTOOL_MSG_PHC_VCLOCKS_GET - do */
+void ethtool_phc_vclocks_get_req_free(struct ethtool_phc_vclocks_get_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+void ethtool_phc_vclocks_get_rsp_free(struct ethtool_phc_vclocks_get_rsp *rsp)
+{
+       ethtool_header_free(&rsp->header);
+       free(rsp);
+}
+
+int ethtool_phc_vclocks_get_rsp_parse(const struct nlmsghdr *nlh, void *data)
+{
+       struct ethtool_phc_vclocks_get_rsp *dst;
+       struct ynl_parse_arg *yarg = data;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+
+       dst = yarg->data;
+       parg.ys = yarg->ys;
+
+       mnl_attr_for_each(attr, nlh, sizeof(struct genlmsghdr)) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_PHC_VCLOCKS_HEADER) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.header = 1;
+
+                       parg.rsp_policy = &ethtool_header_nest;
+                       parg.data = &dst->header;
+                       if (ethtool_header_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_PHC_VCLOCKS_NUM) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.num = 1;
+                       dst->num = mnl_attr_get_u32(attr);
+               }
+       }
+
+       return MNL_CB_OK;
+}
+
+struct ethtool_phc_vclocks_get_rsp *
+ethtool_phc_vclocks_get(struct ynl_sock *ys,
+                       struct ethtool_phc_vclocks_get_req *req)
+{
+       struct ynl_req_state yrs = { .yarg = { .ys = ys, }, };
+       struct ethtool_phc_vclocks_get_rsp *rsp;
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_PHC_VCLOCKS_GET, 1);
+       ys->req_policy = &ethtool_phc_vclocks_nest;
+       yrs.yarg.rsp_policy = &ethtool_phc_vclocks_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_PHC_VCLOCKS_HEADER, &req->header);
+
+       rsp = calloc(1, sizeof(*rsp));
+       yrs.yarg.data = rsp;
+       yrs.cb = ethtool_phc_vclocks_get_rsp_parse;
+       yrs.rsp_cmd = 34;
+
+       err = ynl_exec(ys, nlh, &yrs);
+       if (err < 0)
+               goto err_free;
+
+       return rsp;
+
+err_free:
+       ethtool_phc_vclocks_get_rsp_free(rsp);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_PHC_VCLOCKS_GET - dump */
+void
+ethtool_phc_vclocks_get_list_free(struct ethtool_phc_vclocks_get_list *rsp)
+{
+       struct ethtool_phc_vclocks_get_list *next = rsp;
+
+       while ((void *)next != YNL_LIST_END) {
+               rsp = next;
+               next = rsp->next;
+
+               ethtool_header_free(&rsp->obj.header);
+               free(rsp);
+       }
+}
+
+struct ethtool_phc_vclocks_get_list *
+ethtool_phc_vclocks_get_dump(struct ynl_sock *ys,
+                            struct ethtool_phc_vclocks_get_req_dump *req)
+{
+       struct ynl_dump_state yds = {};
+       struct nlmsghdr *nlh;
+       int err;
+
+       yds.ys = ys;
+       yds.alloc_sz = sizeof(struct ethtool_phc_vclocks_get_list);
+       yds.cb = ethtool_phc_vclocks_get_rsp_parse;
+       yds.rsp_cmd = 34;
+       yds.rsp_policy = &ethtool_phc_vclocks_nest;
+
+       nlh = ynl_gemsg_start_dump(ys, ys->family_id, ETHTOOL_MSG_PHC_VCLOCKS_GET, 1);
+       ys->req_policy = &ethtool_phc_vclocks_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_PHC_VCLOCKS_HEADER, &req->header);
+
+       err = ynl_exec_dump(ys, nlh, &yds);
+       if (err < 0)
+               goto free_list;
+
+       return yds.first;
+
+free_list:
+       ethtool_phc_vclocks_get_list_free(yds.first);
+       return NULL;
+}
+
+/* ============== ETHTOOL_MSG_MODULE_GET ============== */
+/* ETHTOOL_MSG_MODULE_GET - do */
+void ethtool_module_get_req_free(struct ethtool_module_get_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+void ethtool_module_get_rsp_free(struct ethtool_module_get_rsp *rsp)
+{
+       ethtool_header_free(&rsp->header);
+       free(rsp);
+}
+
+int ethtool_module_get_rsp_parse(const struct nlmsghdr *nlh, void *data)
+{
+       struct ethtool_module_get_rsp *dst;
+       struct ynl_parse_arg *yarg = data;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+
+       dst = yarg->data;
+       parg.ys = yarg->ys;
+
+       mnl_attr_for_each(attr, nlh, sizeof(struct genlmsghdr)) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_MODULE_HEADER) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.header = 1;
+
+                       parg.rsp_policy = &ethtool_header_nest;
+                       parg.data = &dst->header;
+                       if (ethtool_header_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_MODULE_POWER_MODE_POLICY) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.power_mode_policy = 1;
+                       dst->power_mode_policy = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_MODULE_POWER_MODE) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.power_mode = 1;
+                       dst->power_mode = mnl_attr_get_u8(attr);
+               }
+       }
+
+       return MNL_CB_OK;
+}
+
+struct ethtool_module_get_rsp *
+ethtool_module_get(struct ynl_sock *ys, struct ethtool_module_get_req *req)
+{
+       struct ynl_req_state yrs = { .yarg = { .ys = ys, }, };
+       struct ethtool_module_get_rsp *rsp;
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_MODULE_GET, 1);
+       ys->req_policy = &ethtool_module_nest;
+       yrs.yarg.rsp_policy = &ethtool_module_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_MODULE_HEADER, &req->header);
+
+       rsp = calloc(1, sizeof(*rsp));
+       yrs.yarg.data = rsp;
+       yrs.cb = ethtool_module_get_rsp_parse;
+       yrs.rsp_cmd = 35;
+
+       err = ynl_exec(ys, nlh, &yrs);
+       if (err < 0)
+               goto err_free;
+
+       return rsp;
+
+err_free:
+       ethtool_module_get_rsp_free(rsp);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_MODULE_GET - dump */
+void ethtool_module_get_list_free(struct ethtool_module_get_list *rsp)
+{
+       struct ethtool_module_get_list *next = rsp;
+
+       while ((void *)next != YNL_LIST_END) {
+               rsp = next;
+               next = rsp->next;
+
+               ethtool_header_free(&rsp->obj.header);
+               free(rsp);
+       }
+}
+
+struct ethtool_module_get_list *
+ethtool_module_get_dump(struct ynl_sock *ys,
+                       struct ethtool_module_get_req_dump *req)
+{
+       struct ynl_dump_state yds = {};
+       struct nlmsghdr *nlh;
+       int err;
+
+       yds.ys = ys;
+       yds.alloc_sz = sizeof(struct ethtool_module_get_list);
+       yds.cb = ethtool_module_get_rsp_parse;
+       yds.rsp_cmd = 35;
+       yds.rsp_policy = &ethtool_module_nest;
+
+       nlh = ynl_gemsg_start_dump(ys, ys->family_id, ETHTOOL_MSG_MODULE_GET, 1);
+       ys->req_policy = &ethtool_module_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_MODULE_HEADER, &req->header);
+
+       err = ynl_exec_dump(ys, nlh, &yds);
+       if (err < 0)
+               goto free_list;
+
+       return yds.first;
+
+free_list:
+       ethtool_module_get_list_free(yds.first);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_MODULE_GET - notify */
+void ethtool_module_get_ntf_free(struct ethtool_module_get_ntf *rsp)
+{
+       ethtool_header_free(&rsp->obj.header);
+       free(rsp);
+}
+
+/* ============== ETHTOOL_MSG_MODULE_SET ============== */
+/* ETHTOOL_MSG_MODULE_SET - do */
+void ethtool_module_set_req_free(struct ethtool_module_set_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+int ethtool_module_set(struct ynl_sock *ys, struct ethtool_module_set_req *req)
+{
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_MODULE_SET, 1);
+       ys->req_policy = &ethtool_module_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_MODULE_HEADER, &req->header);
+       if (req->_present.power_mode_policy)
+               mnl_attr_put_u8(nlh, ETHTOOL_A_MODULE_POWER_MODE_POLICY, req->power_mode_policy);
+       if (req->_present.power_mode)
+               mnl_attr_put_u8(nlh, ETHTOOL_A_MODULE_POWER_MODE, req->power_mode);
+
+       err = ynl_exec(ys, nlh, NULL);
+       if (err < 0)
+               return -1;
+
+       return 0;
+}
+
+/* ============== ETHTOOL_MSG_PSE_GET ============== */
+/* ETHTOOL_MSG_PSE_GET - do */
+void ethtool_pse_get_req_free(struct ethtool_pse_get_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+void ethtool_pse_get_rsp_free(struct ethtool_pse_get_rsp *rsp)
+{
+       ethtool_header_free(&rsp->header);
+       free(rsp);
+}
+
+int ethtool_pse_get_rsp_parse(const struct nlmsghdr *nlh, void *data)
+{
+       struct ynl_parse_arg *yarg = data;
+       struct ethtool_pse_get_rsp *dst;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+
+       dst = yarg->data;
+       parg.ys = yarg->ys;
+
+       mnl_attr_for_each(attr, nlh, sizeof(struct genlmsghdr)) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_PSE_HEADER) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.header = 1;
+
+                       parg.rsp_policy = &ethtool_header_nest;
+                       parg.data = &dst->header;
+                       if (ethtool_header_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_PODL_PSE_ADMIN_STATE) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.admin_state = 1;
+                       dst->admin_state = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_PODL_PSE_ADMIN_CONTROL) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.admin_control = 1;
+                       dst->admin_control = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_PODL_PSE_PW_D_STATUS) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.pw_d_status = 1;
+                       dst->pw_d_status = mnl_attr_get_u32(attr);
+               }
+       }
+
+       return MNL_CB_OK;
+}
+
+struct ethtool_pse_get_rsp *
+ethtool_pse_get(struct ynl_sock *ys, struct ethtool_pse_get_req *req)
+{
+       struct ynl_req_state yrs = { .yarg = { .ys = ys, }, };
+       struct ethtool_pse_get_rsp *rsp;
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_PSE_GET, 1);
+       ys->req_policy = &ethtool_pse_nest;
+       yrs.yarg.rsp_policy = &ethtool_pse_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_PSE_HEADER, &req->header);
+
+       rsp = calloc(1, sizeof(*rsp));
+       yrs.yarg.data = rsp;
+       yrs.cb = ethtool_pse_get_rsp_parse;
+       yrs.rsp_cmd = 37;
+
+       err = ynl_exec(ys, nlh, &yrs);
+       if (err < 0)
+               goto err_free;
+
+       return rsp;
+
+err_free:
+       ethtool_pse_get_rsp_free(rsp);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_PSE_GET - dump */
+void ethtool_pse_get_list_free(struct ethtool_pse_get_list *rsp)
+{
+       struct ethtool_pse_get_list *next = rsp;
+
+       while ((void *)next != YNL_LIST_END) {
+               rsp = next;
+               next = rsp->next;
+
+               ethtool_header_free(&rsp->obj.header);
+               free(rsp);
+       }
+}
+
+struct ethtool_pse_get_list *
+ethtool_pse_get_dump(struct ynl_sock *ys, struct ethtool_pse_get_req_dump *req)
+{
+       struct ynl_dump_state yds = {};
+       struct nlmsghdr *nlh;
+       int err;
+
+       yds.ys = ys;
+       yds.alloc_sz = sizeof(struct ethtool_pse_get_list);
+       yds.cb = ethtool_pse_get_rsp_parse;
+       yds.rsp_cmd = 37;
+       yds.rsp_policy = &ethtool_pse_nest;
+
+       nlh = ynl_gemsg_start_dump(ys, ys->family_id, ETHTOOL_MSG_PSE_GET, 1);
+       ys->req_policy = &ethtool_pse_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_PSE_HEADER, &req->header);
+
+       err = ynl_exec_dump(ys, nlh, &yds);
+       if (err < 0)
+               goto free_list;
+
+       return yds.first;
+
+free_list:
+       ethtool_pse_get_list_free(yds.first);
+       return NULL;
+}
+
+/* ============== ETHTOOL_MSG_PSE_SET ============== */
+/* ETHTOOL_MSG_PSE_SET - do */
+void ethtool_pse_set_req_free(struct ethtool_pse_set_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+int ethtool_pse_set(struct ynl_sock *ys, struct ethtool_pse_set_req *req)
+{
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_PSE_SET, 1);
+       ys->req_policy = &ethtool_pse_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_PSE_HEADER, &req->header);
+       if (req->_present.admin_state)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_PODL_PSE_ADMIN_STATE, req->admin_state);
+       if (req->_present.admin_control)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_PODL_PSE_ADMIN_CONTROL, req->admin_control);
+       if (req->_present.pw_d_status)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_PODL_PSE_PW_D_STATUS, req->pw_d_status);
+
+       err = ynl_exec(ys, nlh, NULL);
+       if (err < 0)
+               return -1;
+
+       return 0;
+}
+
+/* ============== ETHTOOL_MSG_RSS_GET ============== */
+/* ETHTOOL_MSG_RSS_GET - do */
+void ethtool_rss_get_req_free(struct ethtool_rss_get_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+void ethtool_rss_get_rsp_free(struct ethtool_rss_get_rsp *rsp)
+{
+       ethtool_header_free(&rsp->header);
+       free(rsp->indir);
+       free(rsp->hkey);
+       free(rsp);
+}
+
+int ethtool_rss_get_rsp_parse(const struct nlmsghdr *nlh, void *data)
+{
+       struct ynl_parse_arg *yarg = data;
+       struct ethtool_rss_get_rsp *dst;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+
+       dst = yarg->data;
+       parg.ys = yarg->ys;
+
+       mnl_attr_for_each(attr, nlh, sizeof(struct genlmsghdr)) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_RSS_HEADER) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.header = 1;
+
+                       parg.rsp_policy = &ethtool_header_nest;
+                       parg.data = &dst->header;
+                       if (ethtool_header_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_RSS_CONTEXT) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.context = 1;
+                       dst->context = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_RSS_HFUNC) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.hfunc = 1;
+                       dst->hfunc = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_RSS_INDIR) {
+                       unsigned int len;
+
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+
+                       len = mnl_attr_get_payload_len(attr);
+                       dst->_present.indir_len = len;
+                       dst->indir = malloc(len);
+                       memcpy(dst->indir, mnl_attr_get_payload(attr), len);
+               } else if (type == ETHTOOL_A_RSS_HKEY) {
+                       unsigned int len;
+
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+
+                       len = mnl_attr_get_payload_len(attr);
+                       dst->_present.hkey_len = len;
+                       dst->hkey = malloc(len);
+                       memcpy(dst->hkey, mnl_attr_get_payload(attr), len);
+               }
+       }
+
+       return MNL_CB_OK;
+}
+
+struct ethtool_rss_get_rsp *
+ethtool_rss_get(struct ynl_sock *ys, struct ethtool_rss_get_req *req)
+{
+       struct ynl_req_state yrs = { .yarg = { .ys = ys, }, };
+       struct ethtool_rss_get_rsp *rsp;
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_RSS_GET, 1);
+       ys->req_policy = &ethtool_rss_nest;
+       yrs.yarg.rsp_policy = &ethtool_rss_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_RSS_HEADER, &req->header);
+
+       rsp = calloc(1, sizeof(*rsp));
+       yrs.yarg.data = rsp;
+       yrs.cb = ethtool_rss_get_rsp_parse;
+       yrs.rsp_cmd = ETHTOOL_MSG_RSS_GET;
+
+       err = ynl_exec(ys, nlh, &yrs);
+       if (err < 0)
+               goto err_free;
+
+       return rsp;
+
+err_free:
+       ethtool_rss_get_rsp_free(rsp);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_RSS_GET - dump */
+void ethtool_rss_get_list_free(struct ethtool_rss_get_list *rsp)
+{
+       struct ethtool_rss_get_list *next = rsp;
+
+       while ((void *)next != YNL_LIST_END) {
+               rsp = next;
+               next = rsp->next;
+
+               ethtool_header_free(&rsp->obj.header);
+               free(rsp->obj.indir);
+               free(rsp->obj.hkey);
+               free(rsp);
+       }
+}
+
+struct ethtool_rss_get_list *
+ethtool_rss_get_dump(struct ynl_sock *ys, struct ethtool_rss_get_req_dump *req)
+{
+       struct ynl_dump_state yds = {};
+       struct nlmsghdr *nlh;
+       int err;
+
+       yds.ys = ys;
+       yds.alloc_sz = sizeof(struct ethtool_rss_get_list);
+       yds.cb = ethtool_rss_get_rsp_parse;
+       yds.rsp_cmd = ETHTOOL_MSG_RSS_GET;
+       yds.rsp_policy = &ethtool_rss_nest;
+
+       nlh = ynl_gemsg_start_dump(ys, ys->family_id, ETHTOOL_MSG_RSS_GET, 1);
+       ys->req_policy = &ethtool_rss_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_RSS_HEADER, &req->header);
+
+       err = ynl_exec_dump(ys, nlh, &yds);
+       if (err < 0)
+               goto free_list;
+
+       return yds.first;
+
+free_list:
+       ethtool_rss_get_list_free(yds.first);
+       return NULL;
+}
+
+/* ============== ETHTOOL_MSG_PLCA_GET_CFG ============== */
+/* ETHTOOL_MSG_PLCA_GET_CFG - do */
+void ethtool_plca_get_cfg_req_free(struct ethtool_plca_get_cfg_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+void ethtool_plca_get_cfg_rsp_free(struct ethtool_plca_get_cfg_rsp *rsp)
+{
+       ethtool_header_free(&rsp->header);
+       free(rsp);
+}
+
+int ethtool_plca_get_cfg_rsp_parse(const struct nlmsghdr *nlh, void *data)
+{
+       struct ethtool_plca_get_cfg_rsp *dst;
+       struct ynl_parse_arg *yarg = data;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+
+       dst = yarg->data;
+       parg.ys = yarg->ys;
+
+       mnl_attr_for_each(attr, nlh, sizeof(struct genlmsghdr)) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_PLCA_HEADER) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.header = 1;
+
+                       parg.rsp_policy = &ethtool_header_nest;
+                       parg.data = &dst->header;
+                       if (ethtool_header_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_PLCA_VERSION) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.version = 1;
+                       dst->version = mnl_attr_get_u16(attr);
+               } else if (type == ETHTOOL_A_PLCA_ENABLED) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.enabled = 1;
+                       dst->enabled = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_PLCA_STATUS) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.status = 1;
+                       dst->status = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_PLCA_NODE_CNT) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.node_cnt = 1;
+                       dst->node_cnt = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_PLCA_NODE_ID) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.node_id = 1;
+                       dst->node_id = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_PLCA_TO_TMR) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.to_tmr = 1;
+                       dst->to_tmr = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_PLCA_BURST_CNT) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.burst_cnt = 1;
+                       dst->burst_cnt = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_PLCA_BURST_TMR) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.burst_tmr = 1;
+                       dst->burst_tmr = mnl_attr_get_u32(attr);
+               }
+       }
+
+       return MNL_CB_OK;
+}
+
+struct ethtool_plca_get_cfg_rsp *
+ethtool_plca_get_cfg(struct ynl_sock *ys, struct ethtool_plca_get_cfg_req *req)
+{
+       struct ynl_req_state yrs = { .yarg = { .ys = ys, }, };
+       struct ethtool_plca_get_cfg_rsp *rsp;
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_PLCA_GET_CFG, 1);
+       ys->req_policy = &ethtool_plca_nest;
+       yrs.yarg.rsp_policy = &ethtool_plca_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_PLCA_HEADER, &req->header);
+
+       rsp = calloc(1, sizeof(*rsp));
+       yrs.yarg.data = rsp;
+       yrs.cb = ethtool_plca_get_cfg_rsp_parse;
+       yrs.rsp_cmd = ETHTOOL_MSG_PLCA_GET_CFG;
+
+       err = ynl_exec(ys, nlh, &yrs);
+       if (err < 0)
+               goto err_free;
+
+       return rsp;
+
+err_free:
+       ethtool_plca_get_cfg_rsp_free(rsp);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_PLCA_GET_CFG - dump */
+void ethtool_plca_get_cfg_list_free(struct ethtool_plca_get_cfg_list *rsp)
+{
+       struct ethtool_plca_get_cfg_list *next = rsp;
+
+       while ((void *)next != YNL_LIST_END) {
+               rsp = next;
+               next = rsp->next;
+
+               ethtool_header_free(&rsp->obj.header);
+               free(rsp);
+       }
+}
+
+struct ethtool_plca_get_cfg_list *
+ethtool_plca_get_cfg_dump(struct ynl_sock *ys,
+                         struct ethtool_plca_get_cfg_req_dump *req)
+{
+       struct ynl_dump_state yds = {};
+       struct nlmsghdr *nlh;
+       int err;
+
+       yds.ys = ys;
+       yds.alloc_sz = sizeof(struct ethtool_plca_get_cfg_list);
+       yds.cb = ethtool_plca_get_cfg_rsp_parse;
+       yds.rsp_cmd = ETHTOOL_MSG_PLCA_GET_CFG;
+       yds.rsp_policy = &ethtool_plca_nest;
+
+       nlh = ynl_gemsg_start_dump(ys, ys->family_id, ETHTOOL_MSG_PLCA_GET_CFG, 1);
+       ys->req_policy = &ethtool_plca_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_PLCA_HEADER, &req->header);
+
+       err = ynl_exec_dump(ys, nlh, &yds);
+       if (err < 0)
+               goto free_list;
+
+       return yds.first;
+
+free_list:
+       ethtool_plca_get_cfg_list_free(yds.first);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_PLCA_GET_CFG - notify */
+void ethtool_plca_get_cfg_ntf_free(struct ethtool_plca_get_cfg_ntf *rsp)
+{
+       ethtool_header_free(&rsp->obj.header);
+       free(rsp);
+}
+
+/* ============== ETHTOOL_MSG_PLCA_SET_CFG ============== */
+/* ETHTOOL_MSG_PLCA_SET_CFG - do */
+void ethtool_plca_set_cfg_req_free(struct ethtool_plca_set_cfg_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+int ethtool_plca_set_cfg(struct ynl_sock *ys,
+                        struct ethtool_plca_set_cfg_req *req)
+{
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_PLCA_SET_CFG, 1);
+       ys->req_policy = &ethtool_plca_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_PLCA_HEADER, &req->header);
+       if (req->_present.version)
+               mnl_attr_put_u16(nlh, ETHTOOL_A_PLCA_VERSION, req->version);
+       if (req->_present.enabled)
+               mnl_attr_put_u8(nlh, ETHTOOL_A_PLCA_ENABLED, req->enabled);
+       if (req->_present.status)
+               mnl_attr_put_u8(nlh, ETHTOOL_A_PLCA_STATUS, req->status);
+       if (req->_present.node_cnt)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_PLCA_NODE_CNT, req->node_cnt);
+       if (req->_present.node_id)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_PLCA_NODE_ID, req->node_id);
+       if (req->_present.to_tmr)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_PLCA_TO_TMR, req->to_tmr);
+       if (req->_present.burst_cnt)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_PLCA_BURST_CNT, req->burst_cnt);
+       if (req->_present.burst_tmr)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_PLCA_BURST_TMR, req->burst_tmr);
+
+       err = ynl_exec(ys, nlh, NULL);
+       if (err < 0)
+               return -1;
+
+       return 0;
+}
+
+/* ============== ETHTOOL_MSG_PLCA_GET_STATUS ============== */
+/* ETHTOOL_MSG_PLCA_GET_STATUS - do */
+void ethtool_plca_get_status_req_free(struct ethtool_plca_get_status_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+void ethtool_plca_get_status_rsp_free(struct ethtool_plca_get_status_rsp *rsp)
+{
+       ethtool_header_free(&rsp->header);
+       free(rsp);
+}
+
+int ethtool_plca_get_status_rsp_parse(const struct nlmsghdr *nlh, void *data)
+{
+       struct ethtool_plca_get_status_rsp *dst;
+       struct ynl_parse_arg *yarg = data;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+
+       dst = yarg->data;
+       parg.ys = yarg->ys;
+
+       mnl_attr_for_each(attr, nlh, sizeof(struct genlmsghdr)) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_PLCA_HEADER) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.header = 1;
+
+                       parg.rsp_policy = &ethtool_header_nest;
+                       parg.data = &dst->header;
+                       if (ethtool_header_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_PLCA_VERSION) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.version = 1;
+                       dst->version = mnl_attr_get_u16(attr);
+               } else if (type == ETHTOOL_A_PLCA_ENABLED) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.enabled = 1;
+                       dst->enabled = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_PLCA_STATUS) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.status = 1;
+                       dst->status = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_PLCA_NODE_CNT) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.node_cnt = 1;
+                       dst->node_cnt = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_PLCA_NODE_ID) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.node_id = 1;
+                       dst->node_id = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_PLCA_TO_TMR) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.to_tmr = 1;
+                       dst->to_tmr = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_PLCA_BURST_CNT) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.burst_cnt = 1;
+                       dst->burst_cnt = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_PLCA_BURST_TMR) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.burst_tmr = 1;
+                       dst->burst_tmr = mnl_attr_get_u32(attr);
+               }
+       }
+
+       return MNL_CB_OK;
+}
+
+struct ethtool_plca_get_status_rsp *
+ethtool_plca_get_status(struct ynl_sock *ys,
+                       struct ethtool_plca_get_status_req *req)
+{
+       struct ynl_req_state yrs = { .yarg = { .ys = ys, }, };
+       struct ethtool_plca_get_status_rsp *rsp;
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_PLCA_GET_STATUS, 1);
+       ys->req_policy = &ethtool_plca_nest;
+       yrs.yarg.rsp_policy = &ethtool_plca_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_PLCA_HEADER, &req->header);
+
+       rsp = calloc(1, sizeof(*rsp));
+       yrs.yarg.data = rsp;
+       yrs.cb = ethtool_plca_get_status_rsp_parse;
+       yrs.rsp_cmd = 40;
+
+       err = ynl_exec(ys, nlh, &yrs);
+       if (err < 0)
+               goto err_free;
+
+       return rsp;
+
+err_free:
+       ethtool_plca_get_status_rsp_free(rsp);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_PLCA_GET_STATUS - dump */
+void
+ethtool_plca_get_status_list_free(struct ethtool_plca_get_status_list *rsp)
+{
+       struct ethtool_plca_get_status_list *next = rsp;
+
+       while ((void *)next != YNL_LIST_END) {
+               rsp = next;
+               next = rsp->next;
+
+               ethtool_header_free(&rsp->obj.header);
+               free(rsp);
+       }
+}
+
+struct ethtool_plca_get_status_list *
+ethtool_plca_get_status_dump(struct ynl_sock *ys,
+                            struct ethtool_plca_get_status_req_dump *req)
+{
+       struct ynl_dump_state yds = {};
+       struct nlmsghdr *nlh;
+       int err;
+
+       yds.ys = ys;
+       yds.alloc_sz = sizeof(struct ethtool_plca_get_status_list);
+       yds.cb = ethtool_plca_get_status_rsp_parse;
+       yds.rsp_cmd = 40;
+       yds.rsp_policy = &ethtool_plca_nest;
+
+       nlh = ynl_gemsg_start_dump(ys, ys->family_id, ETHTOOL_MSG_PLCA_GET_STATUS, 1);
+       ys->req_policy = &ethtool_plca_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_PLCA_HEADER, &req->header);
+
+       err = ynl_exec_dump(ys, nlh, &yds);
+       if (err < 0)
+               goto free_list;
+
+       return yds.first;
+
+free_list:
+       ethtool_plca_get_status_list_free(yds.first);
+       return NULL;
+}
+
+/* ============== ETHTOOL_MSG_MM_GET ============== */
+/* ETHTOOL_MSG_MM_GET - do */
+void ethtool_mm_get_req_free(struct ethtool_mm_get_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+void ethtool_mm_get_rsp_free(struct ethtool_mm_get_rsp *rsp)
+{
+       ethtool_header_free(&rsp->header);
+       ethtool_mm_stat_free(&rsp->stats);
+       free(rsp);
+}
+
+int ethtool_mm_get_rsp_parse(const struct nlmsghdr *nlh, void *data)
+{
+       struct ynl_parse_arg *yarg = data;
+       struct ethtool_mm_get_rsp *dst;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+
+       dst = yarg->data;
+       parg.ys = yarg->ys;
+
+       mnl_attr_for_each(attr, nlh, sizeof(struct genlmsghdr)) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_MM_HEADER) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.header = 1;
+
+                       parg.rsp_policy = &ethtool_header_nest;
+                       parg.data = &dst->header;
+                       if (ethtool_header_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_MM_PMAC_ENABLED) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.pmac_enabled = 1;
+                       dst->pmac_enabled = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_MM_TX_ENABLED) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.tx_enabled = 1;
+                       dst->tx_enabled = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_MM_TX_ACTIVE) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.tx_active = 1;
+                       dst->tx_active = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_MM_TX_MIN_FRAG_SIZE) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.tx_min_frag_size = 1;
+                       dst->tx_min_frag_size = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_MM_RX_MIN_FRAG_SIZE) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.rx_min_frag_size = 1;
+                       dst->rx_min_frag_size = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_MM_VERIFY_ENABLED) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.verify_enabled = 1;
+                       dst->verify_enabled = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_MM_VERIFY_TIME) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.verify_time = 1;
+                       dst->verify_time = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_MM_MAX_VERIFY_TIME) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.max_verify_time = 1;
+                       dst->max_verify_time = mnl_attr_get_u32(attr);
+               } else if (type == ETHTOOL_A_MM_STATS) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.stats = 1;
+
+                       parg.rsp_policy = &ethtool_mm_stat_nest;
+                       parg.data = &dst->stats;
+                       if (ethtool_mm_stat_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               }
+       }
+
+       return MNL_CB_OK;
+}
+
+struct ethtool_mm_get_rsp *
+ethtool_mm_get(struct ynl_sock *ys, struct ethtool_mm_get_req *req)
+{
+       struct ynl_req_state yrs = { .yarg = { .ys = ys, }, };
+       struct ethtool_mm_get_rsp *rsp;
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_MM_GET, 1);
+       ys->req_policy = &ethtool_mm_nest;
+       yrs.yarg.rsp_policy = &ethtool_mm_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_MM_HEADER, &req->header);
+
+       rsp = calloc(1, sizeof(*rsp));
+       yrs.yarg.data = rsp;
+       yrs.cb = ethtool_mm_get_rsp_parse;
+       yrs.rsp_cmd = ETHTOOL_MSG_MM_GET;
+
+       err = ynl_exec(ys, nlh, &yrs);
+       if (err < 0)
+               goto err_free;
+
+       return rsp;
+
+err_free:
+       ethtool_mm_get_rsp_free(rsp);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_MM_GET - dump */
+void ethtool_mm_get_list_free(struct ethtool_mm_get_list *rsp)
+{
+       struct ethtool_mm_get_list *next = rsp;
+
+       while ((void *)next != YNL_LIST_END) {
+               rsp = next;
+               next = rsp->next;
+
+               ethtool_header_free(&rsp->obj.header);
+               ethtool_mm_stat_free(&rsp->obj.stats);
+               free(rsp);
+       }
+}
+
+struct ethtool_mm_get_list *
+ethtool_mm_get_dump(struct ynl_sock *ys, struct ethtool_mm_get_req_dump *req)
+{
+       struct ynl_dump_state yds = {};
+       struct nlmsghdr *nlh;
+       int err;
+
+       yds.ys = ys;
+       yds.alloc_sz = sizeof(struct ethtool_mm_get_list);
+       yds.cb = ethtool_mm_get_rsp_parse;
+       yds.rsp_cmd = ETHTOOL_MSG_MM_GET;
+       yds.rsp_policy = &ethtool_mm_nest;
+
+       nlh = ynl_gemsg_start_dump(ys, ys->family_id, ETHTOOL_MSG_MM_GET, 1);
+       ys->req_policy = &ethtool_mm_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_MM_HEADER, &req->header);
+
+       err = ynl_exec_dump(ys, nlh, &yds);
+       if (err < 0)
+               goto free_list;
+
+       return yds.first;
+
+free_list:
+       ethtool_mm_get_list_free(yds.first);
+       return NULL;
+}
+
+/* ETHTOOL_MSG_MM_GET - notify */
+void ethtool_mm_get_ntf_free(struct ethtool_mm_get_ntf *rsp)
+{
+       ethtool_header_free(&rsp->obj.header);
+       ethtool_mm_stat_free(&rsp->obj.stats);
+       free(rsp);
+}
+
+/* ============== ETHTOOL_MSG_MM_SET ============== */
+/* ETHTOOL_MSG_MM_SET - do */
+void ethtool_mm_set_req_free(struct ethtool_mm_set_req *req)
+{
+       ethtool_header_free(&req->header);
+       free(req);
+}
+
+int ethtool_mm_set(struct ynl_sock *ys, struct ethtool_mm_set_req *req)
+{
+       struct nlmsghdr *nlh;
+       int err;
+
+       nlh = ynl_gemsg_start_req(ys, ys->family_id, ETHTOOL_MSG_MM_SET, 1);
+       ys->req_policy = &ethtool_mm_nest;
+
+       if (req->_present.header)
+               ethtool_header_put(nlh, ETHTOOL_A_MM_HEADER, &req->header);
+       if (req->_present.verify_enabled)
+               mnl_attr_put_u8(nlh, ETHTOOL_A_MM_VERIFY_ENABLED, req->verify_enabled);
+       if (req->_present.verify_time)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_MM_VERIFY_TIME, req->verify_time);
+       if (req->_present.tx_enabled)
+               mnl_attr_put_u8(nlh, ETHTOOL_A_MM_TX_ENABLED, req->tx_enabled);
+       if (req->_present.pmac_enabled)
+               mnl_attr_put_u8(nlh, ETHTOOL_A_MM_PMAC_ENABLED, req->pmac_enabled);
+       if (req->_present.tx_min_frag_size)
+               mnl_attr_put_u32(nlh, ETHTOOL_A_MM_TX_MIN_FRAG_SIZE, req->tx_min_frag_size);
+
+       err = ynl_exec(ys, nlh, NULL);
+       if (err < 0)
+               return -1;
+
+       return 0;
+}
+
+/* ETHTOOL_MSG_CABLE_TEST_NTF - event */
+int ethtool_cable_test_ntf_rsp_parse(const struct nlmsghdr *nlh, void *data)
+{
+       struct ethtool_cable_test_ntf_rsp *dst;
+       struct ynl_parse_arg *yarg = data;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+
+       dst = yarg->data;
+       parg.ys = yarg->ys;
+
+       mnl_attr_for_each(attr, nlh, sizeof(struct genlmsghdr)) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_CABLE_TEST_NTF_HEADER) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.header = 1;
+
+                       parg.rsp_policy = &ethtool_header_nest;
+                       parg.data = &dst->header;
+                       if (ethtool_header_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_CABLE_TEST_NTF_STATUS) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.status = 1;
+                       dst->status = mnl_attr_get_u8(attr);
+               }
+       }
+
+       return MNL_CB_OK;
+}
+
+void ethtool_cable_test_ntf_free(struct ethtool_cable_test_ntf *rsp)
+{
+       ethtool_header_free(&rsp->obj.header);
+       free(rsp);
+}
+
+/* ETHTOOL_MSG_CABLE_TEST_TDR_NTF - event */
+int ethtool_cable_test_tdr_ntf_rsp_parse(const struct nlmsghdr *nlh,
+                                        void *data)
+{
+       struct ethtool_cable_test_tdr_ntf_rsp *dst;
+       struct ynl_parse_arg *yarg = data;
+       const struct nlattr *attr;
+       struct ynl_parse_arg parg;
+
+       dst = yarg->data;
+       parg.ys = yarg->ys;
+
+       mnl_attr_for_each(attr, nlh, sizeof(struct genlmsghdr)) {
+               unsigned int type = mnl_attr_get_type(attr);
+
+               if (type == ETHTOOL_A_CABLE_TEST_TDR_NTF_HEADER) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.header = 1;
+
+                       parg.rsp_policy = &ethtool_header_nest;
+                       parg.data = &dst->header;
+                       if (ethtool_header_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               } else if (type == ETHTOOL_A_CABLE_TEST_TDR_NTF_STATUS) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.status = 1;
+                       dst->status = mnl_attr_get_u8(attr);
+               } else if (type == ETHTOOL_A_CABLE_TEST_TDR_NTF_NEST) {
+                       if (ynl_attr_validate(yarg, attr))
+                               return MNL_CB_ERROR;
+                       dst->_present.nest = 1;
+
+                       parg.rsp_policy = &ethtool_cable_nest_nest;
+                       parg.data = &dst->nest;
+                       if (ethtool_cable_nest_parse(&parg, attr))
+                               return MNL_CB_ERROR;
+               }
+       }
+
+       return MNL_CB_OK;
+}
+
+void ethtool_cable_test_tdr_ntf_free(struct ethtool_cable_test_tdr_ntf *rsp)
+{
+       ethtool_header_free(&rsp->obj.header);
+       ethtool_cable_nest_free(&rsp->obj.nest);
+       free(rsp);
+}
+
+static const struct ynl_ntf_info ethtool_ntf_info[] =  {
+       [ETHTOOL_MSG_LINKINFO_NTF] =  {
+               .alloc_sz       = sizeof(struct ethtool_linkinfo_get_ntf),
+               .cb             = ethtool_linkinfo_get_rsp_parse,
+               .policy         = &ethtool_linkinfo_nest,
+               .free           = (void *)ethtool_linkinfo_get_ntf_free,
+       },
+       [ETHTOOL_MSG_LINKMODES_NTF] =  {
+               .alloc_sz       = sizeof(struct ethtool_linkmodes_get_ntf),
+               .cb             = ethtool_linkmodes_get_rsp_parse,
+               .policy         = &ethtool_linkmodes_nest,
+               .free           = (void *)ethtool_linkmodes_get_ntf_free,
+       },
+       [ETHTOOL_MSG_DEBUG_NTF] =  {
+               .alloc_sz       = sizeof(struct ethtool_debug_get_ntf),
+               .cb             = ethtool_debug_get_rsp_parse,
+               .policy         = &ethtool_debug_nest,
+               .free           = (void *)ethtool_debug_get_ntf_free,
+       },
+       [ETHTOOL_MSG_WOL_NTF] =  {
+               .alloc_sz       = sizeof(struct ethtool_wol_get_ntf),
+               .cb             = ethtool_wol_get_rsp_parse,
+               .policy         = &ethtool_wol_nest,
+               .free           = (void *)ethtool_wol_get_ntf_free,
+       },
+       [ETHTOOL_MSG_FEATURES_NTF] =  {
+               .alloc_sz       = sizeof(struct ethtool_features_get_ntf),
+               .cb             = ethtool_features_get_rsp_parse,
+               .policy         = &ethtool_features_nest,
+               .free           = (void *)ethtool_features_get_ntf_free,
+       },
+       [ETHTOOL_MSG_PRIVFLAGS_NTF] =  {
+               .alloc_sz       = sizeof(struct ethtool_privflags_get_ntf),
+               .cb             = ethtool_privflags_get_rsp_parse,
+               .policy         = &ethtool_privflags_nest,
+               .free           = (void *)ethtool_privflags_get_ntf_free,
+       },
+       [ETHTOOL_MSG_RINGS_NTF] =  {
+               .alloc_sz       = sizeof(struct ethtool_rings_get_ntf),
+               .cb             = ethtool_rings_get_rsp_parse,
+               .policy         = &ethtool_rings_nest,
+               .free           = (void *)ethtool_rings_get_ntf_free,
+       },
+       [ETHTOOL_MSG_CHANNELS_NTF] =  {
+               .alloc_sz       = sizeof(struct ethtool_channels_get_ntf),
+               .cb             = ethtool_channels_get_rsp_parse,
+               .policy         = &ethtool_channels_nest,
+               .free           = (void *)ethtool_channels_get_ntf_free,
+       },
+       [ETHTOOL_MSG_COALESCE_NTF] =  {
+               .alloc_sz       = sizeof(struct ethtool_coalesce_get_ntf),
+               .cb             = ethtool_coalesce_get_rsp_parse,
+               .policy         = &ethtool_coalesce_nest,
+               .free           = (void *)ethtool_coalesce_get_ntf_free,
+       },
+       [ETHTOOL_MSG_PAUSE_NTF] =  {
+               .alloc_sz       = sizeof(struct ethtool_pause_get_ntf),
+               .cb             = ethtool_pause_get_rsp_parse,
+               .policy         = &ethtool_pause_nest,
+               .free           = (void *)ethtool_pause_get_ntf_free,
+       },
+       [ETHTOOL_MSG_EEE_NTF] =  {
+               .alloc_sz       = sizeof(struct ethtool_eee_get_ntf),
+               .cb             = ethtool_eee_get_rsp_parse,
+               .policy         = &ethtool_eee_nest,
+               .free           = (void *)ethtool_eee_get_ntf_free,
+       },
+       [ETHTOOL_MSG_CABLE_TEST_NTF] =  {
+               .alloc_sz       = sizeof(struct ethtool_cable_test_ntf),
+               .cb             = ethtool_cable_test_ntf_rsp_parse,
+               .policy         = &ethtool_cable_test_ntf_nest,
+               .free           = (void *)ethtool_cable_test_ntf_free,
+       },
+       [ETHTOOL_MSG_CABLE_TEST_TDR_NTF] =  {
+               .alloc_sz       = sizeof(struct ethtool_cable_test_tdr_ntf),
+               .cb             = ethtool_cable_test_tdr_ntf_rsp_parse,
+               .policy         = &ethtool_cable_test_tdr_ntf_nest,
+               .free           = (void *)ethtool_cable_test_tdr_ntf_free,
+       },
+       [ETHTOOL_MSG_FEC_NTF] =  {
+               .alloc_sz       = sizeof(struct ethtool_fec_get_ntf),
+               .cb             = ethtool_fec_get_rsp_parse,
+               .policy         = &ethtool_fec_nest,
+               .free           = (void *)ethtool_fec_get_ntf_free,
+       },
+       [ETHTOOL_MSG_MODULE_NTF] =  {
+               .alloc_sz       = sizeof(struct ethtool_module_get_ntf),
+               .cb             = ethtool_module_get_rsp_parse,
+               .policy         = &ethtool_module_nest,
+               .free           = (void *)ethtool_module_get_ntf_free,
+       },
+       [ETHTOOL_MSG_PLCA_NTF] =  {
+               .alloc_sz       = sizeof(struct ethtool_plca_get_cfg_ntf),
+               .cb             = ethtool_plca_get_cfg_rsp_parse,
+               .policy         = &ethtool_plca_nest,
+               .free           = (void *)ethtool_plca_get_cfg_ntf_free,
+       },
+       [ETHTOOL_MSG_MM_NTF] =  {
+               .alloc_sz       = sizeof(struct ethtool_mm_get_ntf),
+               .cb             = ethtool_mm_get_rsp_parse,
+               .policy         = &ethtool_mm_nest,
+               .free           = (void *)ethtool_mm_get_ntf_free,
+       },
+};
+
+const struct ynl_family ynl_ethtool_family =  {
+       .name           = "ethtool",
+       .ntf_info       = ethtool_ntf_info,
+       .ntf_info_size  = MNL_ARRAY_SIZE(ethtool_ntf_info),
+};
diff --git a/tools/net/ynl/generated/ethtool-user.h b/tools/net/ynl/generated/ethtool-user.h
new file mode 100644 (file)
index 0000000..d7d4ba8
--- /dev/null
@@ -0,0 +1,5531 @@
+/* SPDX-License-Identifier: ((GPL-2.0 WITH Linux-syscall-note) OR BSD-3-Clause) */
+/* Do not edit directly, auto-generated from: */
+/*     Documentation/netlink/specs/ethtool.yaml */
+/* YNL-GEN user header */
+/* YNL-ARG --user-header linux/ethtool_netlink.h --exclude-op stats-get */
+
+#ifndef _LINUX_ETHTOOL_GEN_H
+#define _LINUX_ETHTOOL_GEN_H
+
+#include <stdlib.h>
+#include <string.h>
+#include <linux/types.h>
+#include <linux/ethtool.h>
+
+struct ynl_sock;
+
+extern const struct ynl_family ynl_ethtool_family;
+
+/* Enums */
+const char *ethtool_op_str(int op);
+const char *ethtool_udp_tunnel_type_str(int value);
+const char *ethtool_stringset_str(enum ethtool_stringset value);
+
+/* Common nested types */
+struct ethtool_header {
+       struct {
+               __u32 dev_index:1;
+               __u32 dev_name_len;
+               __u32 flags:1;
+       } _present;
+
+       __u32 dev_index;
+       char *dev_name;
+       __u32 flags;
+};
+
+struct ethtool_pause_stat {
+       struct {
+               __u32 tx_frames:1;
+               __u32 rx_frames:1;
+       } _present;
+
+       __u64 tx_frames;
+       __u64 rx_frames;
+};
+
+struct ethtool_cable_test_tdr_cfg {
+       struct {
+               __u32 first:1;
+               __u32 last:1;
+               __u32 step:1;
+               __u32 pair:1;
+       } _present;
+
+       __u32 first;
+       __u32 last;
+       __u32 step;
+       __u8 pair;
+};
+
+struct ethtool_fec_stat {
+       struct {
+               __u32 corrected_len;
+               __u32 uncorr_len;
+               __u32 corr_bits_len;
+       } _present;
+
+       void *corrected;
+       void *uncorr;
+       void *corr_bits;
+};
+
+struct ethtool_mm_stat {
+       struct {
+               __u32 reassembly_errors:1;
+               __u32 smd_errors:1;
+               __u32 reassembly_ok:1;
+               __u32 rx_frag_count:1;
+               __u32 tx_frag_count:1;
+               __u32 hold_count:1;
+       } _present;
+
+       __u64 reassembly_errors;
+       __u64 smd_errors;
+       __u64 reassembly_ok;
+       __u64 rx_frag_count;
+       __u64 tx_frag_count;
+       __u64 hold_count;
+};
+
+struct ethtool_cable_result {
+       struct {
+               __u32 pair:1;
+               __u32 code:1;
+       } _present;
+
+       __u8 pair;
+       __u8 code;
+};
+
+struct ethtool_cable_fault_length {
+       struct {
+               __u32 pair:1;
+               __u32 cm:1;
+       } _present;
+
+       __u8 pair;
+       __u32 cm;
+};
+
+struct ethtool_bitset_bit {
+       struct {
+               __u32 index:1;
+               __u32 name_len;
+               __u32 value:1;
+       } _present;
+
+       __u32 index;
+       char *name;
+};
+
+struct ethtool_tunnel_udp_entry {
+       struct {
+               __u32 port:1;
+               __u32 type:1;
+       } _present;
+
+       __u16 port /* big-endian */;
+       __u32 type;
+};
+
+struct ethtool_string {
+       struct {
+               __u32 index:1;
+               __u32 value_len;
+       } _present;
+
+       __u32 index;
+       char *value;
+};
+
+struct ethtool_cable_nest {
+       struct {
+               __u32 result:1;
+               __u32 fault_length:1;
+       } _present;
+
+       struct ethtool_cable_result result;
+       struct ethtool_cable_fault_length fault_length;
+};
+
+struct ethtool_bitset_bits {
+       unsigned int n_bit;
+       struct ethtool_bitset_bit *bit;
+};
+
+struct ethtool_strings {
+       unsigned int n_string;
+       struct ethtool_string *string;
+};
+
+struct ethtool_bitset {
+       struct {
+               __u32 nomask:1;
+               __u32 size:1;
+               __u32 bits:1;
+       } _present;
+
+       __u32 size;
+       struct ethtool_bitset_bits bits;
+};
+
+struct ethtool_stringset_ {
+       struct {
+               __u32 id:1;
+               __u32 count:1;
+       } _present;
+
+       __u32 id;
+       __u32 count;
+       unsigned int n_strings;
+       struct ethtool_strings *strings;
+};
+
+struct ethtool_tunnel_udp_table {
+       struct {
+               __u32 size:1;
+               __u32 types:1;
+       } _present;
+
+       __u32 size;
+       struct ethtool_bitset types;
+       unsigned int n_entry;
+       struct ethtool_tunnel_udp_entry *entry;
+};
+
+struct ethtool_stringsets {
+       unsigned int n_stringset;
+       struct ethtool_stringset_ *stringset;
+};
+
+struct ethtool_tunnel_udp {
+       struct {
+               __u32 table:1;
+       } _present;
+
+       struct ethtool_tunnel_udp_table table;
+};
+
+/* ============== ETHTOOL_MSG_STRSET_GET ============== */
+/* ETHTOOL_MSG_STRSET_GET - do */
+struct ethtool_strset_get_req {
+       struct {
+               __u32 header:1;
+               __u32 stringsets:1;
+               __u32 counts_only:1;
+       } _present;
+
+       struct ethtool_header header;
+       struct ethtool_stringsets stringsets;
+};
+
+static inline struct ethtool_strset_get_req *ethtool_strset_get_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_strset_get_req));
+}
+void ethtool_strset_get_req_free(struct ethtool_strset_get_req *req);
+
+static inline void
+ethtool_strset_get_req_set_header_dev_index(struct ethtool_strset_get_req *req,
+                                           __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_strset_get_req_set_header_dev_name(struct ethtool_strset_get_req *req,
+                                          const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_strset_get_req_set_header_flags(struct ethtool_strset_get_req *req,
+                                       __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+static inline void
+__ethtool_strset_get_req_set_stringsets_stringset(struct ethtool_strset_get_req *req,
+                                                 struct ethtool_stringset_ *stringset,
+                                                 unsigned int n_stringset)
+{
+       free(req->stringsets.stringset);
+       req->stringsets.stringset = stringset;
+       req->stringsets.n_stringset = n_stringset;
+}
+static inline void
+ethtool_strset_get_req_set_counts_only(struct ethtool_strset_get_req *req)
+{
+       req->_present.counts_only = 1;
+}
+
+struct ethtool_strset_get_rsp {
+       struct {
+               __u32 header:1;
+               __u32 stringsets:1;
+       } _present;
+
+       struct ethtool_header header;
+       struct ethtool_stringsets stringsets;
+};
+
+void ethtool_strset_get_rsp_free(struct ethtool_strset_get_rsp *rsp);
+
+/*
+ * Get string set from the kernel.
+ */
+struct ethtool_strset_get_rsp *
+ethtool_strset_get(struct ynl_sock *ys, struct ethtool_strset_get_req *req);
+
+/* ETHTOOL_MSG_STRSET_GET - dump */
+struct ethtool_strset_get_req_dump {
+       struct {
+               __u32 header:1;
+               __u32 stringsets:1;
+               __u32 counts_only:1;
+       } _present;
+
+       struct ethtool_header header;
+       struct ethtool_stringsets stringsets;
+};
+
+static inline struct ethtool_strset_get_req_dump *
+ethtool_strset_get_req_dump_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_strset_get_req_dump));
+}
+void ethtool_strset_get_req_dump_free(struct ethtool_strset_get_req_dump *req);
+
+static inline void
+ethtool_strset_get_req_dump_set_header_dev_index(struct ethtool_strset_get_req_dump *req,
+                                                __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_strset_get_req_dump_set_header_dev_name(struct ethtool_strset_get_req_dump *req,
+                                               const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_strset_get_req_dump_set_header_flags(struct ethtool_strset_get_req_dump *req,
+                                            __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+static inline void
+__ethtool_strset_get_req_dump_set_stringsets_stringset(struct ethtool_strset_get_req_dump *req,
+                                                      struct ethtool_stringset_ *stringset,
+                                                      unsigned int n_stringset)
+{
+       free(req->stringsets.stringset);
+       req->stringsets.stringset = stringset;
+       req->stringsets.n_stringset = n_stringset;
+}
+static inline void
+ethtool_strset_get_req_dump_set_counts_only(struct ethtool_strset_get_req_dump *req)
+{
+       req->_present.counts_only = 1;
+}
+
+struct ethtool_strset_get_list {
+       struct ethtool_strset_get_list *next;
+       struct ethtool_strset_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_strset_get_list_free(struct ethtool_strset_get_list *rsp);
+
+struct ethtool_strset_get_list *
+ethtool_strset_get_dump(struct ynl_sock *ys,
+                       struct ethtool_strset_get_req_dump *req);
+
+/* ============== ETHTOOL_MSG_LINKINFO_GET ============== */
+/* ETHTOOL_MSG_LINKINFO_GET - do */
+struct ethtool_linkinfo_get_req {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_linkinfo_get_req *
+ethtool_linkinfo_get_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_linkinfo_get_req));
+}
+void ethtool_linkinfo_get_req_free(struct ethtool_linkinfo_get_req *req);
+
+static inline void
+ethtool_linkinfo_get_req_set_header_dev_index(struct ethtool_linkinfo_get_req *req,
+                                             __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_linkinfo_get_req_set_header_dev_name(struct ethtool_linkinfo_get_req *req,
+                                            const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_linkinfo_get_req_set_header_flags(struct ethtool_linkinfo_get_req *req,
+                                         __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_linkinfo_get_rsp {
+       struct {
+               __u32 header:1;
+               __u32 port:1;
+               __u32 phyaddr:1;
+               __u32 tp_mdix:1;
+               __u32 tp_mdix_ctrl:1;
+               __u32 transceiver:1;
+       } _present;
+
+       struct ethtool_header header;
+       __u8 port;
+       __u8 phyaddr;
+       __u8 tp_mdix;
+       __u8 tp_mdix_ctrl;
+       __u8 transceiver;
+};
+
+void ethtool_linkinfo_get_rsp_free(struct ethtool_linkinfo_get_rsp *rsp);
+
+/*
+ * Get link info.
+ */
+struct ethtool_linkinfo_get_rsp *
+ethtool_linkinfo_get(struct ynl_sock *ys, struct ethtool_linkinfo_get_req *req);
+
+/* ETHTOOL_MSG_LINKINFO_GET - dump */
+struct ethtool_linkinfo_get_req_dump {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_linkinfo_get_req_dump *
+ethtool_linkinfo_get_req_dump_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_linkinfo_get_req_dump));
+}
+void
+ethtool_linkinfo_get_req_dump_free(struct ethtool_linkinfo_get_req_dump *req);
+
+static inline void
+ethtool_linkinfo_get_req_dump_set_header_dev_index(struct ethtool_linkinfo_get_req_dump *req,
+                                                  __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_linkinfo_get_req_dump_set_header_dev_name(struct ethtool_linkinfo_get_req_dump *req,
+                                                 const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_linkinfo_get_req_dump_set_header_flags(struct ethtool_linkinfo_get_req_dump *req,
+                                              __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_linkinfo_get_list {
+       struct ethtool_linkinfo_get_list *next;
+       struct ethtool_linkinfo_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_linkinfo_get_list_free(struct ethtool_linkinfo_get_list *rsp);
+
+struct ethtool_linkinfo_get_list *
+ethtool_linkinfo_get_dump(struct ynl_sock *ys,
+                         struct ethtool_linkinfo_get_req_dump *req);
+
+/* ETHTOOL_MSG_LINKINFO_GET - notify */
+struct ethtool_linkinfo_get_ntf {
+       __u16 family;
+       __u8 cmd;
+       struct ynl_ntf_base_type *next;
+       void (*free)(struct ethtool_linkinfo_get_ntf *ntf);
+       struct ethtool_linkinfo_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_linkinfo_get_ntf_free(struct ethtool_linkinfo_get_ntf *rsp);
+
+/* ============== ETHTOOL_MSG_LINKINFO_SET ============== */
+/* ETHTOOL_MSG_LINKINFO_SET - do */
+struct ethtool_linkinfo_set_req {
+       struct {
+               __u32 header:1;
+               __u32 port:1;
+               __u32 phyaddr:1;
+               __u32 tp_mdix:1;
+               __u32 tp_mdix_ctrl:1;
+               __u32 transceiver:1;
+       } _present;
+
+       struct ethtool_header header;
+       __u8 port;
+       __u8 phyaddr;
+       __u8 tp_mdix;
+       __u8 tp_mdix_ctrl;
+       __u8 transceiver;
+};
+
+static inline struct ethtool_linkinfo_set_req *
+ethtool_linkinfo_set_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_linkinfo_set_req));
+}
+void ethtool_linkinfo_set_req_free(struct ethtool_linkinfo_set_req *req);
+
+static inline void
+ethtool_linkinfo_set_req_set_header_dev_index(struct ethtool_linkinfo_set_req *req,
+                                             __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_linkinfo_set_req_set_header_dev_name(struct ethtool_linkinfo_set_req *req,
+                                            const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_linkinfo_set_req_set_header_flags(struct ethtool_linkinfo_set_req *req,
+                                         __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+static inline void
+ethtool_linkinfo_set_req_set_port(struct ethtool_linkinfo_set_req *req,
+                                 __u8 port)
+{
+       req->_present.port = 1;
+       req->port = port;
+}
+static inline void
+ethtool_linkinfo_set_req_set_phyaddr(struct ethtool_linkinfo_set_req *req,
+                                    __u8 phyaddr)
+{
+       req->_present.phyaddr = 1;
+       req->phyaddr = phyaddr;
+}
+static inline void
+ethtool_linkinfo_set_req_set_tp_mdix(struct ethtool_linkinfo_set_req *req,
+                                    __u8 tp_mdix)
+{
+       req->_present.tp_mdix = 1;
+       req->tp_mdix = tp_mdix;
+}
+static inline void
+ethtool_linkinfo_set_req_set_tp_mdix_ctrl(struct ethtool_linkinfo_set_req *req,
+                                         __u8 tp_mdix_ctrl)
+{
+       req->_present.tp_mdix_ctrl = 1;
+       req->tp_mdix_ctrl = tp_mdix_ctrl;
+}
+static inline void
+ethtool_linkinfo_set_req_set_transceiver(struct ethtool_linkinfo_set_req *req,
+                                        __u8 transceiver)
+{
+       req->_present.transceiver = 1;
+       req->transceiver = transceiver;
+}
+
+/*
+ * Set link info.
+ */
+int ethtool_linkinfo_set(struct ynl_sock *ys,
+                        struct ethtool_linkinfo_set_req *req);
+
+/* ============== ETHTOOL_MSG_LINKMODES_GET ============== */
+/* ETHTOOL_MSG_LINKMODES_GET - do */
+struct ethtool_linkmodes_get_req {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_linkmodes_get_req *
+ethtool_linkmodes_get_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_linkmodes_get_req));
+}
+void ethtool_linkmodes_get_req_free(struct ethtool_linkmodes_get_req *req);
+
+static inline void
+ethtool_linkmodes_get_req_set_header_dev_index(struct ethtool_linkmodes_get_req *req,
+                                              __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_linkmodes_get_req_set_header_dev_name(struct ethtool_linkmodes_get_req *req,
+                                             const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_linkmodes_get_req_set_header_flags(struct ethtool_linkmodes_get_req *req,
+                                          __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_linkmodes_get_rsp {
+       struct {
+               __u32 header:1;
+               __u32 autoneg:1;
+               __u32 ours:1;
+               __u32 peer:1;
+               __u32 speed:1;
+               __u32 duplex:1;
+               __u32 master_slave_cfg:1;
+               __u32 master_slave_state:1;
+               __u32 lanes:1;
+               __u32 rate_matching:1;
+       } _present;
+
+       struct ethtool_header header;
+       __u8 autoneg;
+       struct ethtool_bitset ours;
+       struct ethtool_bitset peer;
+       __u32 speed;
+       __u8 duplex;
+       __u8 master_slave_cfg;
+       __u8 master_slave_state;
+       __u32 lanes;
+       __u8 rate_matching;
+};
+
+void ethtool_linkmodes_get_rsp_free(struct ethtool_linkmodes_get_rsp *rsp);
+
+/*
+ * Get link modes.
+ */
+struct ethtool_linkmodes_get_rsp *
+ethtool_linkmodes_get(struct ynl_sock *ys,
+                     struct ethtool_linkmodes_get_req *req);
+
+/* ETHTOOL_MSG_LINKMODES_GET - dump */
+struct ethtool_linkmodes_get_req_dump {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_linkmodes_get_req_dump *
+ethtool_linkmodes_get_req_dump_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_linkmodes_get_req_dump));
+}
+void
+ethtool_linkmodes_get_req_dump_free(struct ethtool_linkmodes_get_req_dump *req);
+
+static inline void
+ethtool_linkmodes_get_req_dump_set_header_dev_index(struct ethtool_linkmodes_get_req_dump *req,
+                                                   __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_linkmodes_get_req_dump_set_header_dev_name(struct ethtool_linkmodes_get_req_dump *req,
+                                                  const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_linkmodes_get_req_dump_set_header_flags(struct ethtool_linkmodes_get_req_dump *req,
+                                               __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_linkmodes_get_list {
+       struct ethtool_linkmodes_get_list *next;
+       struct ethtool_linkmodes_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_linkmodes_get_list_free(struct ethtool_linkmodes_get_list *rsp);
+
+struct ethtool_linkmodes_get_list *
+ethtool_linkmodes_get_dump(struct ynl_sock *ys,
+                          struct ethtool_linkmodes_get_req_dump *req);
+
+/* ETHTOOL_MSG_LINKMODES_GET - notify */
+struct ethtool_linkmodes_get_ntf {
+       __u16 family;
+       __u8 cmd;
+       struct ynl_ntf_base_type *next;
+       void (*free)(struct ethtool_linkmodes_get_ntf *ntf);
+       struct ethtool_linkmodes_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_linkmodes_get_ntf_free(struct ethtool_linkmodes_get_ntf *rsp);
+
+/* ============== ETHTOOL_MSG_LINKMODES_SET ============== */
+/* ETHTOOL_MSG_LINKMODES_SET - do */
+struct ethtool_linkmodes_set_req {
+       struct {
+               __u32 header:1;
+               __u32 autoneg:1;
+               __u32 ours:1;
+               __u32 peer:1;
+               __u32 speed:1;
+               __u32 duplex:1;
+               __u32 master_slave_cfg:1;
+               __u32 master_slave_state:1;
+               __u32 lanes:1;
+               __u32 rate_matching:1;
+       } _present;
+
+       struct ethtool_header header;
+       __u8 autoneg;
+       struct ethtool_bitset ours;
+       struct ethtool_bitset peer;
+       __u32 speed;
+       __u8 duplex;
+       __u8 master_slave_cfg;
+       __u8 master_slave_state;
+       __u32 lanes;
+       __u8 rate_matching;
+};
+
+static inline struct ethtool_linkmodes_set_req *
+ethtool_linkmodes_set_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_linkmodes_set_req));
+}
+void ethtool_linkmodes_set_req_free(struct ethtool_linkmodes_set_req *req);
+
+static inline void
+ethtool_linkmodes_set_req_set_header_dev_index(struct ethtool_linkmodes_set_req *req,
+                                              __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_linkmodes_set_req_set_header_dev_name(struct ethtool_linkmodes_set_req *req,
+                                             const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_linkmodes_set_req_set_header_flags(struct ethtool_linkmodes_set_req *req,
+                                          __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+static inline void
+ethtool_linkmodes_set_req_set_autoneg(struct ethtool_linkmodes_set_req *req,
+                                     __u8 autoneg)
+{
+       req->_present.autoneg = 1;
+       req->autoneg = autoneg;
+}
+static inline void
+ethtool_linkmodes_set_req_set_ours_nomask(struct ethtool_linkmodes_set_req *req)
+{
+       req->_present.ours = 1;
+       req->ours._present.nomask = 1;
+}
+static inline void
+ethtool_linkmodes_set_req_set_ours_size(struct ethtool_linkmodes_set_req *req,
+                                       __u32 size)
+{
+       req->_present.ours = 1;
+       req->ours._present.size = 1;
+       req->ours.size = size;
+}
+static inline void
+__ethtool_linkmodes_set_req_set_ours_bits_bit(struct ethtool_linkmodes_set_req *req,
+                                             struct ethtool_bitset_bit *bit,
+                                             unsigned int n_bit)
+{
+       free(req->ours.bits.bit);
+       req->ours.bits.bit = bit;
+       req->ours.bits.n_bit = n_bit;
+}
+static inline void
+ethtool_linkmodes_set_req_set_peer_nomask(struct ethtool_linkmodes_set_req *req)
+{
+       req->_present.peer = 1;
+       req->peer._present.nomask = 1;
+}
+static inline void
+ethtool_linkmodes_set_req_set_peer_size(struct ethtool_linkmodes_set_req *req,
+                                       __u32 size)
+{
+       req->_present.peer = 1;
+       req->peer._present.size = 1;
+       req->peer.size = size;
+}
+static inline void
+__ethtool_linkmodes_set_req_set_peer_bits_bit(struct ethtool_linkmodes_set_req *req,
+                                             struct ethtool_bitset_bit *bit,
+                                             unsigned int n_bit)
+{
+       free(req->peer.bits.bit);
+       req->peer.bits.bit = bit;
+       req->peer.bits.n_bit = n_bit;
+}
+static inline void
+ethtool_linkmodes_set_req_set_speed(struct ethtool_linkmodes_set_req *req,
+                                   __u32 speed)
+{
+       req->_present.speed = 1;
+       req->speed = speed;
+}
+static inline void
+ethtool_linkmodes_set_req_set_duplex(struct ethtool_linkmodes_set_req *req,
+                                    __u8 duplex)
+{
+       req->_present.duplex = 1;
+       req->duplex = duplex;
+}
+static inline void
+ethtool_linkmodes_set_req_set_master_slave_cfg(struct ethtool_linkmodes_set_req *req,
+                                              __u8 master_slave_cfg)
+{
+       req->_present.master_slave_cfg = 1;
+       req->master_slave_cfg = master_slave_cfg;
+}
+static inline void
+ethtool_linkmodes_set_req_set_master_slave_state(struct ethtool_linkmodes_set_req *req,
+                                                __u8 master_slave_state)
+{
+       req->_present.master_slave_state = 1;
+       req->master_slave_state = master_slave_state;
+}
+static inline void
+ethtool_linkmodes_set_req_set_lanes(struct ethtool_linkmodes_set_req *req,
+                                   __u32 lanes)
+{
+       req->_present.lanes = 1;
+       req->lanes = lanes;
+}
+static inline void
+ethtool_linkmodes_set_req_set_rate_matching(struct ethtool_linkmodes_set_req *req,
+                                           __u8 rate_matching)
+{
+       req->_present.rate_matching = 1;
+       req->rate_matching = rate_matching;
+}
+
+/*
+ * Set link modes.
+ */
+int ethtool_linkmodes_set(struct ynl_sock *ys,
+                         struct ethtool_linkmodes_set_req *req);
+
+/* ============== ETHTOOL_MSG_LINKSTATE_GET ============== */
+/* ETHTOOL_MSG_LINKSTATE_GET - do */
+struct ethtool_linkstate_get_req {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_linkstate_get_req *
+ethtool_linkstate_get_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_linkstate_get_req));
+}
+void ethtool_linkstate_get_req_free(struct ethtool_linkstate_get_req *req);
+
+static inline void
+ethtool_linkstate_get_req_set_header_dev_index(struct ethtool_linkstate_get_req *req,
+                                              __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_linkstate_get_req_set_header_dev_name(struct ethtool_linkstate_get_req *req,
+                                             const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_linkstate_get_req_set_header_flags(struct ethtool_linkstate_get_req *req,
+                                          __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_linkstate_get_rsp {
+       struct {
+               __u32 header:1;
+               __u32 link:1;
+               __u32 sqi:1;
+               __u32 sqi_max:1;
+               __u32 ext_state:1;
+               __u32 ext_substate:1;
+               __u32 ext_down_cnt:1;
+       } _present;
+
+       struct ethtool_header header;
+       __u8 link;
+       __u32 sqi;
+       __u32 sqi_max;
+       __u8 ext_state;
+       __u8 ext_substate;
+       __u32 ext_down_cnt;
+};
+
+void ethtool_linkstate_get_rsp_free(struct ethtool_linkstate_get_rsp *rsp);
+
+/*
+ * Get link state.
+ */
+struct ethtool_linkstate_get_rsp *
+ethtool_linkstate_get(struct ynl_sock *ys,
+                     struct ethtool_linkstate_get_req *req);
+
+/* ETHTOOL_MSG_LINKSTATE_GET - dump */
+struct ethtool_linkstate_get_req_dump {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_linkstate_get_req_dump *
+ethtool_linkstate_get_req_dump_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_linkstate_get_req_dump));
+}
+void
+ethtool_linkstate_get_req_dump_free(struct ethtool_linkstate_get_req_dump *req);
+
+static inline void
+ethtool_linkstate_get_req_dump_set_header_dev_index(struct ethtool_linkstate_get_req_dump *req,
+                                                   __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_linkstate_get_req_dump_set_header_dev_name(struct ethtool_linkstate_get_req_dump *req,
+                                                  const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_linkstate_get_req_dump_set_header_flags(struct ethtool_linkstate_get_req_dump *req,
+                                               __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_linkstate_get_list {
+       struct ethtool_linkstate_get_list *next;
+       struct ethtool_linkstate_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_linkstate_get_list_free(struct ethtool_linkstate_get_list *rsp);
+
+struct ethtool_linkstate_get_list *
+ethtool_linkstate_get_dump(struct ynl_sock *ys,
+                          struct ethtool_linkstate_get_req_dump *req);
+
+/* ============== ETHTOOL_MSG_DEBUG_GET ============== */
+/* ETHTOOL_MSG_DEBUG_GET - do */
+struct ethtool_debug_get_req {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_debug_get_req *ethtool_debug_get_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_debug_get_req));
+}
+void ethtool_debug_get_req_free(struct ethtool_debug_get_req *req);
+
+static inline void
+ethtool_debug_get_req_set_header_dev_index(struct ethtool_debug_get_req *req,
+                                          __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_debug_get_req_set_header_dev_name(struct ethtool_debug_get_req *req,
+                                         const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_debug_get_req_set_header_flags(struct ethtool_debug_get_req *req,
+                                      __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_debug_get_rsp {
+       struct {
+               __u32 header:1;
+               __u32 msgmask:1;
+       } _present;
+
+       struct ethtool_header header;
+       struct ethtool_bitset msgmask;
+};
+
+void ethtool_debug_get_rsp_free(struct ethtool_debug_get_rsp *rsp);
+
+/*
+ * Get debug message mask.
+ */
+struct ethtool_debug_get_rsp *
+ethtool_debug_get(struct ynl_sock *ys, struct ethtool_debug_get_req *req);
+
+/* ETHTOOL_MSG_DEBUG_GET - dump */
+struct ethtool_debug_get_req_dump {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_debug_get_req_dump *
+ethtool_debug_get_req_dump_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_debug_get_req_dump));
+}
+void ethtool_debug_get_req_dump_free(struct ethtool_debug_get_req_dump *req);
+
+static inline void
+ethtool_debug_get_req_dump_set_header_dev_index(struct ethtool_debug_get_req_dump *req,
+                                               __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_debug_get_req_dump_set_header_dev_name(struct ethtool_debug_get_req_dump *req,
+                                              const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_debug_get_req_dump_set_header_flags(struct ethtool_debug_get_req_dump *req,
+                                           __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_debug_get_list {
+       struct ethtool_debug_get_list *next;
+       struct ethtool_debug_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_debug_get_list_free(struct ethtool_debug_get_list *rsp);
+
+struct ethtool_debug_get_list *
+ethtool_debug_get_dump(struct ynl_sock *ys,
+                      struct ethtool_debug_get_req_dump *req);
+
+/* ETHTOOL_MSG_DEBUG_GET - notify */
+struct ethtool_debug_get_ntf {
+       __u16 family;
+       __u8 cmd;
+       struct ynl_ntf_base_type *next;
+       void (*free)(struct ethtool_debug_get_ntf *ntf);
+       struct ethtool_debug_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_debug_get_ntf_free(struct ethtool_debug_get_ntf *rsp);
+
+/* ============== ETHTOOL_MSG_DEBUG_SET ============== */
+/* ETHTOOL_MSG_DEBUG_SET - do */
+struct ethtool_debug_set_req {
+       struct {
+               __u32 header:1;
+               __u32 msgmask:1;
+       } _present;
+
+       struct ethtool_header header;
+       struct ethtool_bitset msgmask;
+};
+
+static inline struct ethtool_debug_set_req *ethtool_debug_set_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_debug_set_req));
+}
+void ethtool_debug_set_req_free(struct ethtool_debug_set_req *req);
+
+static inline void
+ethtool_debug_set_req_set_header_dev_index(struct ethtool_debug_set_req *req,
+                                          __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_debug_set_req_set_header_dev_name(struct ethtool_debug_set_req *req,
+                                         const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_debug_set_req_set_header_flags(struct ethtool_debug_set_req *req,
+                                      __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+static inline void
+ethtool_debug_set_req_set_msgmask_nomask(struct ethtool_debug_set_req *req)
+{
+       req->_present.msgmask = 1;
+       req->msgmask._present.nomask = 1;
+}
+static inline void
+ethtool_debug_set_req_set_msgmask_size(struct ethtool_debug_set_req *req,
+                                      __u32 size)
+{
+       req->_present.msgmask = 1;
+       req->msgmask._present.size = 1;
+       req->msgmask.size = size;
+}
+static inline void
+__ethtool_debug_set_req_set_msgmask_bits_bit(struct ethtool_debug_set_req *req,
+                                            struct ethtool_bitset_bit *bit,
+                                            unsigned int n_bit)
+{
+       free(req->msgmask.bits.bit);
+       req->msgmask.bits.bit = bit;
+       req->msgmask.bits.n_bit = n_bit;
+}
+
+/*
+ * Set debug message mask.
+ */
+int ethtool_debug_set(struct ynl_sock *ys, struct ethtool_debug_set_req *req);
+
+/* ============== ETHTOOL_MSG_WOL_GET ============== */
+/* ETHTOOL_MSG_WOL_GET - do */
+struct ethtool_wol_get_req {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_wol_get_req *ethtool_wol_get_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_wol_get_req));
+}
+void ethtool_wol_get_req_free(struct ethtool_wol_get_req *req);
+
+static inline void
+ethtool_wol_get_req_set_header_dev_index(struct ethtool_wol_get_req *req,
+                                        __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_wol_get_req_set_header_dev_name(struct ethtool_wol_get_req *req,
+                                       const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_wol_get_req_set_header_flags(struct ethtool_wol_get_req *req,
+                                    __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_wol_get_rsp {
+       struct {
+               __u32 header:1;
+               __u32 modes:1;
+               __u32 sopass_len;
+       } _present;
+
+       struct ethtool_header header;
+       struct ethtool_bitset modes;
+       void *sopass;
+};
+
+void ethtool_wol_get_rsp_free(struct ethtool_wol_get_rsp *rsp);
+
+/*
+ * Get WOL params.
+ */
+struct ethtool_wol_get_rsp *
+ethtool_wol_get(struct ynl_sock *ys, struct ethtool_wol_get_req *req);
+
+/* ETHTOOL_MSG_WOL_GET - dump */
+struct ethtool_wol_get_req_dump {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_wol_get_req_dump *
+ethtool_wol_get_req_dump_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_wol_get_req_dump));
+}
+void ethtool_wol_get_req_dump_free(struct ethtool_wol_get_req_dump *req);
+
+static inline void
+ethtool_wol_get_req_dump_set_header_dev_index(struct ethtool_wol_get_req_dump *req,
+                                             __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_wol_get_req_dump_set_header_dev_name(struct ethtool_wol_get_req_dump *req,
+                                            const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_wol_get_req_dump_set_header_flags(struct ethtool_wol_get_req_dump *req,
+                                         __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_wol_get_list {
+       struct ethtool_wol_get_list *next;
+       struct ethtool_wol_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_wol_get_list_free(struct ethtool_wol_get_list *rsp);
+
+struct ethtool_wol_get_list *
+ethtool_wol_get_dump(struct ynl_sock *ys, struct ethtool_wol_get_req_dump *req);
+
+/* ETHTOOL_MSG_WOL_GET - notify */
+struct ethtool_wol_get_ntf {
+       __u16 family;
+       __u8 cmd;
+       struct ynl_ntf_base_type *next;
+       void (*free)(struct ethtool_wol_get_ntf *ntf);
+       struct ethtool_wol_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_wol_get_ntf_free(struct ethtool_wol_get_ntf *rsp);
+
+/* ============== ETHTOOL_MSG_WOL_SET ============== */
+/* ETHTOOL_MSG_WOL_SET - do */
+struct ethtool_wol_set_req {
+       struct {
+               __u32 header:1;
+               __u32 modes:1;
+               __u32 sopass_len;
+       } _present;
+
+       struct ethtool_header header;
+       struct ethtool_bitset modes;
+       void *sopass;
+};
+
+static inline struct ethtool_wol_set_req *ethtool_wol_set_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_wol_set_req));
+}
+void ethtool_wol_set_req_free(struct ethtool_wol_set_req *req);
+
+static inline void
+ethtool_wol_set_req_set_header_dev_index(struct ethtool_wol_set_req *req,
+                                        __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_wol_set_req_set_header_dev_name(struct ethtool_wol_set_req *req,
+                                       const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_wol_set_req_set_header_flags(struct ethtool_wol_set_req *req,
+                                    __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+static inline void
+ethtool_wol_set_req_set_modes_nomask(struct ethtool_wol_set_req *req)
+{
+       req->_present.modes = 1;
+       req->modes._present.nomask = 1;
+}
+static inline void
+ethtool_wol_set_req_set_modes_size(struct ethtool_wol_set_req *req, __u32 size)
+{
+       req->_present.modes = 1;
+       req->modes._present.size = 1;
+       req->modes.size = size;
+}
+static inline void
+__ethtool_wol_set_req_set_modes_bits_bit(struct ethtool_wol_set_req *req,
+                                        struct ethtool_bitset_bit *bit,
+                                        unsigned int n_bit)
+{
+       free(req->modes.bits.bit);
+       req->modes.bits.bit = bit;
+       req->modes.bits.n_bit = n_bit;
+}
+static inline void
+ethtool_wol_set_req_set_sopass(struct ethtool_wol_set_req *req,
+                              const void *sopass, size_t len)
+{
+       free(req->sopass);
+       req->sopass = malloc(req->_present.sopass_len);
+       memcpy(req->sopass, sopass, req->_present.sopass_len);
+}
+
+/*
+ * Set WOL params.
+ */
+int ethtool_wol_set(struct ynl_sock *ys, struct ethtool_wol_set_req *req);
+
+/* ============== ETHTOOL_MSG_FEATURES_GET ============== */
+/* ETHTOOL_MSG_FEATURES_GET - do */
+struct ethtool_features_get_req {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_features_get_req *
+ethtool_features_get_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_features_get_req));
+}
+void ethtool_features_get_req_free(struct ethtool_features_get_req *req);
+
+static inline void
+ethtool_features_get_req_set_header_dev_index(struct ethtool_features_get_req *req,
+                                             __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_features_get_req_set_header_dev_name(struct ethtool_features_get_req *req,
+                                            const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_features_get_req_set_header_flags(struct ethtool_features_get_req *req,
+                                         __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_features_get_rsp {
+       struct {
+               __u32 header:1;
+               __u32 hw:1;
+               __u32 wanted:1;
+               __u32 active:1;
+               __u32 nochange:1;
+       } _present;
+
+       struct ethtool_header header;
+       struct ethtool_bitset hw;
+       struct ethtool_bitset wanted;
+       struct ethtool_bitset active;
+       struct ethtool_bitset nochange;
+};
+
+void ethtool_features_get_rsp_free(struct ethtool_features_get_rsp *rsp);
+
+/*
+ * Get features.
+ */
+struct ethtool_features_get_rsp *
+ethtool_features_get(struct ynl_sock *ys, struct ethtool_features_get_req *req);
+
+/* ETHTOOL_MSG_FEATURES_GET - dump */
+struct ethtool_features_get_req_dump {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_features_get_req_dump *
+ethtool_features_get_req_dump_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_features_get_req_dump));
+}
+void
+ethtool_features_get_req_dump_free(struct ethtool_features_get_req_dump *req);
+
+static inline void
+ethtool_features_get_req_dump_set_header_dev_index(struct ethtool_features_get_req_dump *req,
+                                                  __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_features_get_req_dump_set_header_dev_name(struct ethtool_features_get_req_dump *req,
+                                                 const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_features_get_req_dump_set_header_flags(struct ethtool_features_get_req_dump *req,
+                                              __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_features_get_list {
+       struct ethtool_features_get_list *next;
+       struct ethtool_features_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_features_get_list_free(struct ethtool_features_get_list *rsp);
+
+struct ethtool_features_get_list *
+ethtool_features_get_dump(struct ynl_sock *ys,
+                         struct ethtool_features_get_req_dump *req);
+
+/* ETHTOOL_MSG_FEATURES_GET - notify */
+struct ethtool_features_get_ntf {
+       __u16 family;
+       __u8 cmd;
+       struct ynl_ntf_base_type *next;
+       void (*free)(struct ethtool_features_get_ntf *ntf);
+       struct ethtool_features_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_features_get_ntf_free(struct ethtool_features_get_ntf *rsp);
+
+/* ============== ETHTOOL_MSG_FEATURES_SET ============== */
+/* ETHTOOL_MSG_FEATURES_SET - do */
+struct ethtool_features_set_req {
+       struct {
+               __u32 header:1;
+               __u32 hw:1;
+               __u32 wanted:1;
+               __u32 active:1;
+               __u32 nochange:1;
+       } _present;
+
+       struct ethtool_header header;
+       struct ethtool_bitset hw;
+       struct ethtool_bitset wanted;
+       struct ethtool_bitset active;
+       struct ethtool_bitset nochange;
+};
+
+static inline struct ethtool_features_set_req *
+ethtool_features_set_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_features_set_req));
+}
+void ethtool_features_set_req_free(struct ethtool_features_set_req *req);
+
+static inline void
+ethtool_features_set_req_set_header_dev_index(struct ethtool_features_set_req *req,
+                                             __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_features_set_req_set_header_dev_name(struct ethtool_features_set_req *req,
+                                            const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_features_set_req_set_header_flags(struct ethtool_features_set_req *req,
+                                         __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+static inline void
+ethtool_features_set_req_set_hw_nomask(struct ethtool_features_set_req *req)
+{
+       req->_present.hw = 1;
+       req->hw._present.nomask = 1;
+}
+static inline void
+ethtool_features_set_req_set_hw_size(struct ethtool_features_set_req *req,
+                                    __u32 size)
+{
+       req->_present.hw = 1;
+       req->hw._present.size = 1;
+       req->hw.size = size;
+}
+static inline void
+__ethtool_features_set_req_set_hw_bits_bit(struct ethtool_features_set_req *req,
+                                          struct ethtool_bitset_bit *bit,
+                                          unsigned int n_bit)
+{
+       free(req->hw.bits.bit);
+       req->hw.bits.bit = bit;
+       req->hw.bits.n_bit = n_bit;
+}
+static inline void
+ethtool_features_set_req_set_wanted_nomask(struct ethtool_features_set_req *req)
+{
+       req->_present.wanted = 1;
+       req->wanted._present.nomask = 1;
+}
+static inline void
+ethtool_features_set_req_set_wanted_size(struct ethtool_features_set_req *req,
+                                        __u32 size)
+{
+       req->_present.wanted = 1;
+       req->wanted._present.size = 1;
+       req->wanted.size = size;
+}
+static inline void
+__ethtool_features_set_req_set_wanted_bits_bit(struct ethtool_features_set_req *req,
+                                              struct ethtool_bitset_bit *bit,
+                                              unsigned int n_bit)
+{
+       free(req->wanted.bits.bit);
+       req->wanted.bits.bit = bit;
+       req->wanted.bits.n_bit = n_bit;
+}
+static inline void
+ethtool_features_set_req_set_active_nomask(struct ethtool_features_set_req *req)
+{
+       req->_present.active = 1;
+       req->active._present.nomask = 1;
+}
+static inline void
+ethtool_features_set_req_set_active_size(struct ethtool_features_set_req *req,
+                                        __u32 size)
+{
+       req->_present.active = 1;
+       req->active._present.size = 1;
+       req->active.size = size;
+}
+static inline void
+__ethtool_features_set_req_set_active_bits_bit(struct ethtool_features_set_req *req,
+                                              struct ethtool_bitset_bit *bit,
+                                              unsigned int n_bit)
+{
+       free(req->active.bits.bit);
+       req->active.bits.bit = bit;
+       req->active.bits.n_bit = n_bit;
+}
+static inline void
+ethtool_features_set_req_set_nochange_nomask(struct ethtool_features_set_req *req)
+{
+       req->_present.nochange = 1;
+       req->nochange._present.nomask = 1;
+}
+static inline void
+ethtool_features_set_req_set_nochange_size(struct ethtool_features_set_req *req,
+                                          __u32 size)
+{
+       req->_present.nochange = 1;
+       req->nochange._present.size = 1;
+       req->nochange.size = size;
+}
+static inline void
+__ethtool_features_set_req_set_nochange_bits_bit(struct ethtool_features_set_req *req,
+                                                struct ethtool_bitset_bit *bit,
+                                                unsigned int n_bit)
+{
+       free(req->nochange.bits.bit);
+       req->nochange.bits.bit = bit;
+       req->nochange.bits.n_bit = n_bit;
+}
+
+struct ethtool_features_set_rsp {
+       struct {
+               __u32 header:1;
+               __u32 hw:1;
+               __u32 wanted:1;
+               __u32 active:1;
+               __u32 nochange:1;
+       } _present;
+
+       struct ethtool_header header;
+       struct ethtool_bitset hw;
+       struct ethtool_bitset wanted;
+       struct ethtool_bitset active;
+       struct ethtool_bitset nochange;
+};
+
+void ethtool_features_set_rsp_free(struct ethtool_features_set_rsp *rsp);
+
+/*
+ * Set features.
+ */
+struct ethtool_features_set_rsp *
+ethtool_features_set(struct ynl_sock *ys, struct ethtool_features_set_req *req);
+
+/* ============== ETHTOOL_MSG_PRIVFLAGS_GET ============== */
+/* ETHTOOL_MSG_PRIVFLAGS_GET - do */
+struct ethtool_privflags_get_req {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_privflags_get_req *
+ethtool_privflags_get_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_privflags_get_req));
+}
+void ethtool_privflags_get_req_free(struct ethtool_privflags_get_req *req);
+
+static inline void
+ethtool_privflags_get_req_set_header_dev_index(struct ethtool_privflags_get_req *req,
+                                              __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_privflags_get_req_set_header_dev_name(struct ethtool_privflags_get_req *req,
+                                             const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_privflags_get_req_set_header_flags(struct ethtool_privflags_get_req *req,
+                                          __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_privflags_get_rsp {
+       struct {
+               __u32 header:1;
+               __u32 flags:1;
+       } _present;
+
+       struct ethtool_header header;
+       struct ethtool_bitset flags;
+};
+
+void ethtool_privflags_get_rsp_free(struct ethtool_privflags_get_rsp *rsp);
+
+/*
+ * Get device private flags.
+ */
+struct ethtool_privflags_get_rsp *
+ethtool_privflags_get(struct ynl_sock *ys,
+                     struct ethtool_privflags_get_req *req);
+
+/* ETHTOOL_MSG_PRIVFLAGS_GET - dump */
+struct ethtool_privflags_get_req_dump {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_privflags_get_req_dump *
+ethtool_privflags_get_req_dump_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_privflags_get_req_dump));
+}
+void
+ethtool_privflags_get_req_dump_free(struct ethtool_privflags_get_req_dump *req);
+
+static inline void
+ethtool_privflags_get_req_dump_set_header_dev_index(struct ethtool_privflags_get_req_dump *req,
+                                                   __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_privflags_get_req_dump_set_header_dev_name(struct ethtool_privflags_get_req_dump *req,
+                                                  const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_privflags_get_req_dump_set_header_flags(struct ethtool_privflags_get_req_dump *req,
+                                               __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_privflags_get_list {
+       struct ethtool_privflags_get_list *next;
+       struct ethtool_privflags_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_privflags_get_list_free(struct ethtool_privflags_get_list *rsp);
+
+struct ethtool_privflags_get_list *
+ethtool_privflags_get_dump(struct ynl_sock *ys,
+                          struct ethtool_privflags_get_req_dump *req);
+
+/* ETHTOOL_MSG_PRIVFLAGS_GET - notify */
+struct ethtool_privflags_get_ntf {
+       __u16 family;
+       __u8 cmd;
+       struct ynl_ntf_base_type *next;
+       void (*free)(struct ethtool_privflags_get_ntf *ntf);
+       struct ethtool_privflags_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_privflags_get_ntf_free(struct ethtool_privflags_get_ntf *rsp);
+
+/* ============== ETHTOOL_MSG_PRIVFLAGS_SET ============== */
+/* ETHTOOL_MSG_PRIVFLAGS_SET - do */
+struct ethtool_privflags_set_req {
+       struct {
+               __u32 header:1;
+               __u32 flags:1;
+       } _present;
+
+       struct ethtool_header header;
+       struct ethtool_bitset flags;
+};
+
+static inline struct ethtool_privflags_set_req *
+ethtool_privflags_set_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_privflags_set_req));
+}
+void ethtool_privflags_set_req_free(struct ethtool_privflags_set_req *req);
+
+static inline void
+ethtool_privflags_set_req_set_header_dev_index(struct ethtool_privflags_set_req *req,
+                                              __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_privflags_set_req_set_header_dev_name(struct ethtool_privflags_set_req *req,
+                                             const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_privflags_set_req_set_header_flags(struct ethtool_privflags_set_req *req,
+                                          __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+static inline void
+ethtool_privflags_set_req_set_flags_nomask(struct ethtool_privflags_set_req *req)
+{
+       req->_present.flags = 1;
+       req->flags._present.nomask = 1;
+}
+static inline void
+ethtool_privflags_set_req_set_flags_size(struct ethtool_privflags_set_req *req,
+                                        __u32 size)
+{
+       req->_present.flags = 1;
+       req->flags._present.size = 1;
+       req->flags.size = size;
+}
+static inline void
+__ethtool_privflags_set_req_set_flags_bits_bit(struct ethtool_privflags_set_req *req,
+                                              struct ethtool_bitset_bit *bit,
+                                              unsigned int n_bit)
+{
+       free(req->flags.bits.bit);
+       req->flags.bits.bit = bit;
+       req->flags.bits.n_bit = n_bit;
+}
+
+/*
+ * Set device private flags.
+ */
+int ethtool_privflags_set(struct ynl_sock *ys,
+                         struct ethtool_privflags_set_req *req);
+
+/* ============== ETHTOOL_MSG_RINGS_GET ============== */
+/* ETHTOOL_MSG_RINGS_GET - do */
+struct ethtool_rings_get_req {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_rings_get_req *ethtool_rings_get_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_rings_get_req));
+}
+void ethtool_rings_get_req_free(struct ethtool_rings_get_req *req);
+
+static inline void
+ethtool_rings_get_req_set_header_dev_index(struct ethtool_rings_get_req *req,
+                                          __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_rings_get_req_set_header_dev_name(struct ethtool_rings_get_req *req,
+                                         const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_rings_get_req_set_header_flags(struct ethtool_rings_get_req *req,
+                                      __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_rings_get_rsp {
+       struct {
+               __u32 header:1;
+               __u32 rx_max:1;
+               __u32 rx_mini_max:1;
+               __u32 rx_jumbo_max:1;
+               __u32 tx_max:1;
+               __u32 rx:1;
+               __u32 rx_mini:1;
+               __u32 rx_jumbo:1;
+               __u32 tx:1;
+               __u32 rx_buf_len:1;
+               __u32 tcp_data_split:1;
+               __u32 cqe_size:1;
+               __u32 tx_push:1;
+               __u32 rx_push:1;
+               __u32 tx_push_buf_len:1;
+               __u32 tx_push_buf_len_max:1;
+       } _present;
+
+       struct ethtool_header header;
+       __u32 rx_max;
+       __u32 rx_mini_max;
+       __u32 rx_jumbo_max;
+       __u32 tx_max;
+       __u32 rx;
+       __u32 rx_mini;
+       __u32 rx_jumbo;
+       __u32 tx;
+       __u32 rx_buf_len;
+       __u8 tcp_data_split;
+       __u32 cqe_size;
+       __u8 tx_push;
+       __u8 rx_push;
+       __u32 tx_push_buf_len;
+       __u32 tx_push_buf_len_max;
+};
+
+void ethtool_rings_get_rsp_free(struct ethtool_rings_get_rsp *rsp);
+
+/*
+ * Get ring params.
+ */
+struct ethtool_rings_get_rsp *
+ethtool_rings_get(struct ynl_sock *ys, struct ethtool_rings_get_req *req);
+
+/* ETHTOOL_MSG_RINGS_GET - dump */
+struct ethtool_rings_get_req_dump {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_rings_get_req_dump *
+ethtool_rings_get_req_dump_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_rings_get_req_dump));
+}
+void ethtool_rings_get_req_dump_free(struct ethtool_rings_get_req_dump *req);
+
+static inline void
+ethtool_rings_get_req_dump_set_header_dev_index(struct ethtool_rings_get_req_dump *req,
+                                               __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_rings_get_req_dump_set_header_dev_name(struct ethtool_rings_get_req_dump *req,
+                                              const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_rings_get_req_dump_set_header_flags(struct ethtool_rings_get_req_dump *req,
+                                           __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_rings_get_list {
+       struct ethtool_rings_get_list *next;
+       struct ethtool_rings_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_rings_get_list_free(struct ethtool_rings_get_list *rsp);
+
+struct ethtool_rings_get_list *
+ethtool_rings_get_dump(struct ynl_sock *ys,
+                      struct ethtool_rings_get_req_dump *req);
+
+/* ETHTOOL_MSG_RINGS_GET - notify */
+struct ethtool_rings_get_ntf {
+       __u16 family;
+       __u8 cmd;
+       struct ynl_ntf_base_type *next;
+       void (*free)(struct ethtool_rings_get_ntf *ntf);
+       struct ethtool_rings_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_rings_get_ntf_free(struct ethtool_rings_get_ntf *rsp);
+
+/* ============== ETHTOOL_MSG_RINGS_SET ============== */
+/* ETHTOOL_MSG_RINGS_SET - do */
+struct ethtool_rings_set_req {
+       struct {
+               __u32 header:1;
+               __u32 rx_max:1;
+               __u32 rx_mini_max:1;
+               __u32 rx_jumbo_max:1;
+               __u32 tx_max:1;
+               __u32 rx:1;
+               __u32 rx_mini:1;
+               __u32 rx_jumbo:1;
+               __u32 tx:1;
+               __u32 rx_buf_len:1;
+               __u32 tcp_data_split:1;
+               __u32 cqe_size:1;
+               __u32 tx_push:1;
+               __u32 rx_push:1;
+               __u32 tx_push_buf_len:1;
+               __u32 tx_push_buf_len_max:1;
+       } _present;
+
+       struct ethtool_header header;
+       __u32 rx_max;
+       __u32 rx_mini_max;
+       __u32 rx_jumbo_max;
+       __u32 tx_max;
+       __u32 rx;
+       __u32 rx_mini;
+       __u32 rx_jumbo;
+       __u32 tx;
+       __u32 rx_buf_len;
+       __u8 tcp_data_split;
+       __u32 cqe_size;
+       __u8 tx_push;
+       __u8 rx_push;
+       __u32 tx_push_buf_len;
+       __u32 tx_push_buf_len_max;
+};
+
+static inline struct ethtool_rings_set_req *ethtool_rings_set_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_rings_set_req));
+}
+void ethtool_rings_set_req_free(struct ethtool_rings_set_req *req);
+
+static inline void
+ethtool_rings_set_req_set_header_dev_index(struct ethtool_rings_set_req *req,
+                                          __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_rings_set_req_set_header_dev_name(struct ethtool_rings_set_req *req,
+                                         const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_rings_set_req_set_header_flags(struct ethtool_rings_set_req *req,
+                                      __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+static inline void
+ethtool_rings_set_req_set_rx_max(struct ethtool_rings_set_req *req,
+                                __u32 rx_max)
+{
+       req->_present.rx_max = 1;
+       req->rx_max = rx_max;
+}
+static inline void
+ethtool_rings_set_req_set_rx_mini_max(struct ethtool_rings_set_req *req,
+                                     __u32 rx_mini_max)
+{
+       req->_present.rx_mini_max = 1;
+       req->rx_mini_max = rx_mini_max;
+}
+static inline void
+ethtool_rings_set_req_set_rx_jumbo_max(struct ethtool_rings_set_req *req,
+                                      __u32 rx_jumbo_max)
+{
+       req->_present.rx_jumbo_max = 1;
+       req->rx_jumbo_max = rx_jumbo_max;
+}
+static inline void
+ethtool_rings_set_req_set_tx_max(struct ethtool_rings_set_req *req,
+                                __u32 tx_max)
+{
+       req->_present.tx_max = 1;
+       req->tx_max = tx_max;
+}
+static inline void
+ethtool_rings_set_req_set_rx(struct ethtool_rings_set_req *req, __u32 rx)
+{
+       req->_present.rx = 1;
+       req->rx = rx;
+}
+static inline void
+ethtool_rings_set_req_set_rx_mini(struct ethtool_rings_set_req *req,
+                                 __u32 rx_mini)
+{
+       req->_present.rx_mini = 1;
+       req->rx_mini = rx_mini;
+}
+static inline void
+ethtool_rings_set_req_set_rx_jumbo(struct ethtool_rings_set_req *req,
+                                  __u32 rx_jumbo)
+{
+       req->_present.rx_jumbo = 1;
+       req->rx_jumbo = rx_jumbo;
+}
+static inline void
+ethtool_rings_set_req_set_tx(struct ethtool_rings_set_req *req, __u32 tx)
+{
+       req->_present.tx = 1;
+       req->tx = tx;
+}
+static inline void
+ethtool_rings_set_req_set_rx_buf_len(struct ethtool_rings_set_req *req,
+                                    __u32 rx_buf_len)
+{
+       req->_present.rx_buf_len = 1;
+       req->rx_buf_len = rx_buf_len;
+}
+static inline void
+ethtool_rings_set_req_set_tcp_data_split(struct ethtool_rings_set_req *req,
+                                        __u8 tcp_data_split)
+{
+       req->_present.tcp_data_split = 1;
+       req->tcp_data_split = tcp_data_split;
+}
+static inline void
+ethtool_rings_set_req_set_cqe_size(struct ethtool_rings_set_req *req,
+                                  __u32 cqe_size)
+{
+       req->_present.cqe_size = 1;
+       req->cqe_size = cqe_size;
+}
+static inline void
+ethtool_rings_set_req_set_tx_push(struct ethtool_rings_set_req *req,
+                                 __u8 tx_push)
+{
+       req->_present.tx_push = 1;
+       req->tx_push = tx_push;
+}
+static inline void
+ethtool_rings_set_req_set_rx_push(struct ethtool_rings_set_req *req,
+                                 __u8 rx_push)
+{
+       req->_present.rx_push = 1;
+       req->rx_push = rx_push;
+}
+static inline void
+ethtool_rings_set_req_set_tx_push_buf_len(struct ethtool_rings_set_req *req,
+                                         __u32 tx_push_buf_len)
+{
+       req->_present.tx_push_buf_len = 1;
+       req->tx_push_buf_len = tx_push_buf_len;
+}
+static inline void
+ethtool_rings_set_req_set_tx_push_buf_len_max(struct ethtool_rings_set_req *req,
+                                             __u32 tx_push_buf_len_max)
+{
+       req->_present.tx_push_buf_len_max = 1;
+       req->tx_push_buf_len_max = tx_push_buf_len_max;
+}
+
+/*
+ * Set ring params.
+ */
+int ethtool_rings_set(struct ynl_sock *ys, struct ethtool_rings_set_req *req);
+
+/* ============== ETHTOOL_MSG_CHANNELS_GET ============== */
+/* ETHTOOL_MSG_CHANNELS_GET - do */
+struct ethtool_channels_get_req {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_channels_get_req *
+ethtool_channels_get_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_channels_get_req));
+}
+void ethtool_channels_get_req_free(struct ethtool_channels_get_req *req);
+
+static inline void
+ethtool_channels_get_req_set_header_dev_index(struct ethtool_channels_get_req *req,
+                                             __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_channels_get_req_set_header_dev_name(struct ethtool_channels_get_req *req,
+                                            const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_channels_get_req_set_header_flags(struct ethtool_channels_get_req *req,
+                                         __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_channels_get_rsp {
+       struct {
+               __u32 header:1;
+               __u32 rx_max:1;
+               __u32 tx_max:1;
+               __u32 other_max:1;
+               __u32 combined_max:1;
+               __u32 rx_count:1;
+               __u32 tx_count:1;
+               __u32 other_count:1;
+               __u32 combined_count:1;
+       } _present;
+
+       struct ethtool_header header;
+       __u32 rx_max;
+       __u32 tx_max;
+       __u32 other_max;
+       __u32 combined_max;
+       __u32 rx_count;
+       __u32 tx_count;
+       __u32 other_count;
+       __u32 combined_count;
+};
+
+void ethtool_channels_get_rsp_free(struct ethtool_channels_get_rsp *rsp);
+
+/*
+ * Get channel params.
+ */
+struct ethtool_channels_get_rsp *
+ethtool_channels_get(struct ynl_sock *ys, struct ethtool_channels_get_req *req);
+
+/* ETHTOOL_MSG_CHANNELS_GET - dump */
+struct ethtool_channels_get_req_dump {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_channels_get_req_dump *
+ethtool_channels_get_req_dump_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_channels_get_req_dump));
+}
+void
+ethtool_channels_get_req_dump_free(struct ethtool_channels_get_req_dump *req);
+
+static inline void
+ethtool_channels_get_req_dump_set_header_dev_index(struct ethtool_channels_get_req_dump *req,
+                                                  __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_channels_get_req_dump_set_header_dev_name(struct ethtool_channels_get_req_dump *req,
+                                                 const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_channels_get_req_dump_set_header_flags(struct ethtool_channels_get_req_dump *req,
+                                              __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_channels_get_list {
+       struct ethtool_channels_get_list *next;
+       struct ethtool_channels_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_channels_get_list_free(struct ethtool_channels_get_list *rsp);
+
+struct ethtool_channels_get_list *
+ethtool_channels_get_dump(struct ynl_sock *ys,
+                         struct ethtool_channels_get_req_dump *req);
+
+/* ETHTOOL_MSG_CHANNELS_GET - notify */
+struct ethtool_channels_get_ntf {
+       __u16 family;
+       __u8 cmd;
+       struct ynl_ntf_base_type *next;
+       void (*free)(struct ethtool_channels_get_ntf *ntf);
+       struct ethtool_channels_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_channels_get_ntf_free(struct ethtool_channels_get_ntf *rsp);
+
+/* ============== ETHTOOL_MSG_CHANNELS_SET ============== */
+/* ETHTOOL_MSG_CHANNELS_SET - do */
+struct ethtool_channels_set_req {
+       struct {
+               __u32 header:1;
+               __u32 rx_max:1;
+               __u32 tx_max:1;
+               __u32 other_max:1;
+               __u32 combined_max:1;
+               __u32 rx_count:1;
+               __u32 tx_count:1;
+               __u32 other_count:1;
+               __u32 combined_count:1;
+       } _present;
+
+       struct ethtool_header header;
+       __u32 rx_max;
+       __u32 tx_max;
+       __u32 other_max;
+       __u32 combined_max;
+       __u32 rx_count;
+       __u32 tx_count;
+       __u32 other_count;
+       __u32 combined_count;
+};
+
+static inline struct ethtool_channels_set_req *
+ethtool_channels_set_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_channels_set_req));
+}
+void ethtool_channels_set_req_free(struct ethtool_channels_set_req *req);
+
+static inline void
+ethtool_channels_set_req_set_header_dev_index(struct ethtool_channels_set_req *req,
+                                             __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_channels_set_req_set_header_dev_name(struct ethtool_channels_set_req *req,
+                                            const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_channels_set_req_set_header_flags(struct ethtool_channels_set_req *req,
+                                         __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+static inline void
+ethtool_channels_set_req_set_rx_max(struct ethtool_channels_set_req *req,
+                                   __u32 rx_max)
+{
+       req->_present.rx_max = 1;
+       req->rx_max = rx_max;
+}
+static inline void
+ethtool_channels_set_req_set_tx_max(struct ethtool_channels_set_req *req,
+                                   __u32 tx_max)
+{
+       req->_present.tx_max = 1;
+       req->tx_max = tx_max;
+}
+static inline void
+ethtool_channels_set_req_set_other_max(struct ethtool_channels_set_req *req,
+                                      __u32 other_max)
+{
+       req->_present.other_max = 1;
+       req->other_max = other_max;
+}
+static inline void
+ethtool_channels_set_req_set_combined_max(struct ethtool_channels_set_req *req,
+                                         __u32 combined_max)
+{
+       req->_present.combined_max = 1;
+       req->combined_max = combined_max;
+}
+static inline void
+ethtool_channels_set_req_set_rx_count(struct ethtool_channels_set_req *req,
+                                     __u32 rx_count)
+{
+       req->_present.rx_count = 1;
+       req->rx_count = rx_count;
+}
+static inline void
+ethtool_channels_set_req_set_tx_count(struct ethtool_channels_set_req *req,
+                                     __u32 tx_count)
+{
+       req->_present.tx_count = 1;
+       req->tx_count = tx_count;
+}
+static inline void
+ethtool_channels_set_req_set_other_count(struct ethtool_channels_set_req *req,
+                                        __u32 other_count)
+{
+       req->_present.other_count = 1;
+       req->other_count = other_count;
+}
+static inline void
+ethtool_channels_set_req_set_combined_count(struct ethtool_channels_set_req *req,
+                                           __u32 combined_count)
+{
+       req->_present.combined_count = 1;
+       req->combined_count = combined_count;
+}
+
+/*
+ * Set channel params.
+ */
+int ethtool_channels_set(struct ynl_sock *ys,
+                        struct ethtool_channels_set_req *req);
+
+/* ============== ETHTOOL_MSG_COALESCE_GET ============== */
+/* ETHTOOL_MSG_COALESCE_GET - do */
+struct ethtool_coalesce_get_req {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_coalesce_get_req *
+ethtool_coalesce_get_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_coalesce_get_req));
+}
+void ethtool_coalesce_get_req_free(struct ethtool_coalesce_get_req *req);
+
+static inline void
+ethtool_coalesce_get_req_set_header_dev_index(struct ethtool_coalesce_get_req *req,
+                                             __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_coalesce_get_req_set_header_dev_name(struct ethtool_coalesce_get_req *req,
+                                            const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_coalesce_get_req_set_header_flags(struct ethtool_coalesce_get_req *req,
+                                         __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_coalesce_get_rsp {
+       struct {
+               __u32 header:1;
+               __u32 rx_usecs:1;
+               __u32 rx_max_frames:1;
+               __u32 rx_usecs_irq:1;
+               __u32 rx_max_frames_irq:1;
+               __u32 tx_usecs:1;
+               __u32 tx_max_frames:1;
+               __u32 tx_usecs_irq:1;
+               __u32 tx_max_frames_irq:1;
+               __u32 stats_block_usecs:1;
+               __u32 use_adaptive_rx:1;
+               __u32 use_adaptive_tx:1;
+               __u32 pkt_rate_low:1;
+               __u32 rx_usecs_low:1;
+               __u32 rx_max_frames_low:1;
+               __u32 tx_usecs_low:1;
+               __u32 tx_max_frames_low:1;
+               __u32 pkt_rate_high:1;
+               __u32 rx_usecs_high:1;
+               __u32 rx_max_frames_high:1;
+               __u32 tx_usecs_high:1;
+               __u32 tx_max_frames_high:1;
+               __u32 rate_sample_interval:1;
+               __u32 use_cqe_mode_tx:1;
+               __u32 use_cqe_mode_rx:1;
+               __u32 tx_aggr_max_bytes:1;
+               __u32 tx_aggr_max_frames:1;
+               __u32 tx_aggr_time_usecs:1;
+       } _present;
+
+       struct ethtool_header header;
+       __u32 rx_usecs;
+       __u32 rx_max_frames;
+       __u32 rx_usecs_irq;
+       __u32 rx_max_frames_irq;
+       __u32 tx_usecs;
+       __u32 tx_max_frames;
+       __u32 tx_usecs_irq;
+       __u32 tx_max_frames_irq;
+       __u32 stats_block_usecs;
+       __u8 use_adaptive_rx;
+       __u8 use_adaptive_tx;
+       __u32 pkt_rate_low;
+       __u32 rx_usecs_low;
+       __u32 rx_max_frames_low;
+       __u32 tx_usecs_low;
+       __u32 tx_max_frames_low;
+       __u32 pkt_rate_high;
+       __u32 rx_usecs_high;
+       __u32 rx_max_frames_high;
+       __u32 tx_usecs_high;
+       __u32 tx_max_frames_high;
+       __u32 rate_sample_interval;
+       __u8 use_cqe_mode_tx;
+       __u8 use_cqe_mode_rx;
+       __u32 tx_aggr_max_bytes;
+       __u32 tx_aggr_max_frames;
+       __u32 tx_aggr_time_usecs;
+};
+
+void ethtool_coalesce_get_rsp_free(struct ethtool_coalesce_get_rsp *rsp);
+
+/*
+ * Get coalesce params.
+ */
+struct ethtool_coalesce_get_rsp *
+ethtool_coalesce_get(struct ynl_sock *ys, struct ethtool_coalesce_get_req *req);
+
+/* ETHTOOL_MSG_COALESCE_GET - dump */
+struct ethtool_coalesce_get_req_dump {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_coalesce_get_req_dump *
+ethtool_coalesce_get_req_dump_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_coalesce_get_req_dump));
+}
+void
+ethtool_coalesce_get_req_dump_free(struct ethtool_coalesce_get_req_dump *req);
+
+static inline void
+ethtool_coalesce_get_req_dump_set_header_dev_index(struct ethtool_coalesce_get_req_dump *req,
+                                                  __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_coalesce_get_req_dump_set_header_dev_name(struct ethtool_coalesce_get_req_dump *req,
+                                                 const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_coalesce_get_req_dump_set_header_flags(struct ethtool_coalesce_get_req_dump *req,
+                                              __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_coalesce_get_list {
+       struct ethtool_coalesce_get_list *next;
+       struct ethtool_coalesce_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_coalesce_get_list_free(struct ethtool_coalesce_get_list *rsp);
+
+struct ethtool_coalesce_get_list *
+ethtool_coalesce_get_dump(struct ynl_sock *ys,
+                         struct ethtool_coalesce_get_req_dump *req);
+
+/* ETHTOOL_MSG_COALESCE_GET - notify */
+struct ethtool_coalesce_get_ntf {
+       __u16 family;
+       __u8 cmd;
+       struct ynl_ntf_base_type *next;
+       void (*free)(struct ethtool_coalesce_get_ntf *ntf);
+       struct ethtool_coalesce_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_coalesce_get_ntf_free(struct ethtool_coalesce_get_ntf *rsp);
+
+/* ============== ETHTOOL_MSG_COALESCE_SET ============== */
+/* ETHTOOL_MSG_COALESCE_SET - do */
+struct ethtool_coalesce_set_req {
+       struct {
+               __u32 header:1;
+               __u32 rx_usecs:1;
+               __u32 rx_max_frames:1;
+               __u32 rx_usecs_irq:1;
+               __u32 rx_max_frames_irq:1;
+               __u32 tx_usecs:1;
+               __u32 tx_max_frames:1;
+               __u32 tx_usecs_irq:1;
+               __u32 tx_max_frames_irq:1;
+               __u32 stats_block_usecs:1;
+               __u32 use_adaptive_rx:1;
+               __u32 use_adaptive_tx:1;
+               __u32 pkt_rate_low:1;
+               __u32 rx_usecs_low:1;
+               __u32 rx_max_frames_low:1;
+               __u32 tx_usecs_low:1;
+               __u32 tx_max_frames_low:1;
+               __u32 pkt_rate_high:1;
+               __u32 rx_usecs_high:1;
+               __u32 rx_max_frames_high:1;
+               __u32 tx_usecs_high:1;
+               __u32 tx_max_frames_high:1;
+               __u32 rate_sample_interval:1;
+               __u32 use_cqe_mode_tx:1;
+               __u32 use_cqe_mode_rx:1;
+               __u32 tx_aggr_max_bytes:1;
+               __u32 tx_aggr_max_frames:1;
+               __u32 tx_aggr_time_usecs:1;
+       } _present;
+
+       struct ethtool_header header;
+       __u32 rx_usecs;
+       __u32 rx_max_frames;
+       __u32 rx_usecs_irq;
+       __u32 rx_max_frames_irq;
+       __u32 tx_usecs;
+       __u32 tx_max_frames;
+       __u32 tx_usecs_irq;
+       __u32 tx_max_frames_irq;
+       __u32 stats_block_usecs;
+       __u8 use_adaptive_rx;
+       __u8 use_adaptive_tx;
+       __u32 pkt_rate_low;
+       __u32 rx_usecs_low;
+       __u32 rx_max_frames_low;
+       __u32 tx_usecs_low;
+       __u32 tx_max_frames_low;
+       __u32 pkt_rate_high;
+       __u32 rx_usecs_high;
+       __u32 rx_max_frames_high;
+       __u32 tx_usecs_high;
+       __u32 tx_max_frames_high;
+       __u32 rate_sample_interval;
+       __u8 use_cqe_mode_tx;
+       __u8 use_cqe_mode_rx;
+       __u32 tx_aggr_max_bytes;
+       __u32 tx_aggr_max_frames;
+       __u32 tx_aggr_time_usecs;
+};
+
+static inline struct ethtool_coalesce_set_req *
+ethtool_coalesce_set_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_coalesce_set_req));
+}
+void ethtool_coalesce_set_req_free(struct ethtool_coalesce_set_req *req);
+
+static inline void
+ethtool_coalesce_set_req_set_header_dev_index(struct ethtool_coalesce_set_req *req,
+                                             __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_coalesce_set_req_set_header_dev_name(struct ethtool_coalesce_set_req *req,
+                                            const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_coalesce_set_req_set_header_flags(struct ethtool_coalesce_set_req *req,
+                                         __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+static inline void
+ethtool_coalesce_set_req_set_rx_usecs(struct ethtool_coalesce_set_req *req,
+                                     __u32 rx_usecs)
+{
+       req->_present.rx_usecs = 1;
+       req->rx_usecs = rx_usecs;
+}
+static inline void
+ethtool_coalesce_set_req_set_rx_max_frames(struct ethtool_coalesce_set_req *req,
+                                          __u32 rx_max_frames)
+{
+       req->_present.rx_max_frames = 1;
+       req->rx_max_frames = rx_max_frames;
+}
+static inline void
+ethtool_coalesce_set_req_set_rx_usecs_irq(struct ethtool_coalesce_set_req *req,
+                                         __u32 rx_usecs_irq)
+{
+       req->_present.rx_usecs_irq = 1;
+       req->rx_usecs_irq = rx_usecs_irq;
+}
+static inline void
+ethtool_coalesce_set_req_set_rx_max_frames_irq(struct ethtool_coalesce_set_req *req,
+                                              __u32 rx_max_frames_irq)
+{
+       req->_present.rx_max_frames_irq = 1;
+       req->rx_max_frames_irq = rx_max_frames_irq;
+}
+static inline void
+ethtool_coalesce_set_req_set_tx_usecs(struct ethtool_coalesce_set_req *req,
+                                     __u32 tx_usecs)
+{
+       req->_present.tx_usecs = 1;
+       req->tx_usecs = tx_usecs;
+}
+static inline void
+ethtool_coalesce_set_req_set_tx_max_frames(struct ethtool_coalesce_set_req *req,
+                                          __u32 tx_max_frames)
+{
+       req->_present.tx_max_frames = 1;
+       req->tx_max_frames = tx_max_frames;
+}
+static inline void
+ethtool_coalesce_set_req_set_tx_usecs_irq(struct ethtool_coalesce_set_req *req,
+                                         __u32 tx_usecs_irq)
+{
+       req->_present.tx_usecs_irq = 1;
+       req->tx_usecs_irq = tx_usecs_irq;
+}
+static inline void
+ethtool_coalesce_set_req_set_tx_max_frames_irq(struct ethtool_coalesce_set_req *req,
+                                              __u32 tx_max_frames_irq)
+{
+       req->_present.tx_max_frames_irq = 1;
+       req->tx_max_frames_irq = tx_max_frames_irq;
+}
+static inline void
+ethtool_coalesce_set_req_set_stats_block_usecs(struct ethtool_coalesce_set_req *req,
+                                              __u32 stats_block_usecs)
+{
+       req->_present.stats_block_usecs = 1;
+       req->stats_block_usecs = stats_block_usecs;
+}
+static inline void
+ethtool_coalesce_set_req_set_use_adaptive_rx(struct ethtool_coalesce_set_req *req,
+                                            __u8 use_adaptive_rx)
+{
+       req->_present.use_adaptive_rx = 1;
+       req->use_adaptive_rx = use_adaptive_rx;
+}
+static inline void
+ethtool_coalesce_set_req_set_use_adaptive_tx(struct ethtool_coalesce_set_req *req,
+                                            __u8 use_adaptive_tx)
+{
+       req->_present.use_adaptive_tx = 1;
+       req->use_adaptive_tx = use_adaptive_tx;
+}
+static inline void
+ethtool_coalesce_set_req_set_pkt_rate_low(struct ethtool_coalesce_set_req *req,
+                                         __u32 pkt_rate_low)
+{
+       req->_present.pkt_rate_low = 1;
+       req->pkt_rate_low = pkt_rate_low;
+}
+static inline void
+ethtool_coalesce_set_req_set_rx_usecs_low(struct ethtool_coalesce_set_req *req,
+                                         __u32 rx_usecs_low)
+{
+       req->_present.rx_usecs_low = 1;
+       req->rx_usecs_low = rx_usecs_low;
+}
+static inline void
+ethtool_coalesce_set_req_set_rx_max_frames_low(struct ethtool_coalesce_set_req *req,
+                                              __u32 rx_max_frames_low)
+{
+       req->_present.rx_max_frames_low = 1;
+       req->rx_max_frames_low = rx_max_frames_low;
+}
+static inline void
+ethtool_coalesce_set_req_set_tx_usecs_low(struct ethtool_coalesce_set_req *req,
+                                         __u32 tx_usecs_low)
+{
+       req->_present.tx_usecs_low = 1;
+       req->tx_usecs_low = tx_usecs_low;
+}
+static inline void
+ethtool_coalesce_set_req_set_tx_max_frames_low(struct ethtool_coalesce_set_req *req,
+                                              __u32 tx_max_frames_low)
+{
+       req->_present.tx_max_frames_low = 1;
+       req->tx_max_frames_low = tx_max_frames_low;
+}
+static inline void
+ethtool_coalesce_set_req_set_pkt_rate_high(struct ethtool_coalesce_set_req *req,
+                                          __u32 pkt_rate_high)
+{
+       req->_present.pkt_rate_high = 1;
+       req->pkt_rate_high = pkt_rate_high;
+}
+static inline void
+ethtool_coalesce_set_req_set_rx_usecs_high(struct ethtool_coalesce_set_req *req,
+                                          __u32 rx_usecs_high)
+{
+       req->_present.rx_usecs_high = 1;
+       req->rx_usecs_high = rx_usecs_high;
+}
+static inline void
+ethtool_coalesce_set_req_set_rx_max_frames_high(struct ethtool_coalesce_set_req *req,
+                                               __u32 rx_max_frames_high)
+{
+       req->_present.rx_max_frames_high = 1;
+       req->rx_max_frames_high = rx_max_frames_high;
+}
+static inline void
+ethtool_coalesce_set_req_set_tx_usecs_high(struct ethtool_coalesce_set_req *req,
+                                          __u32 tx_usecs_high)
+{
+       req->_present.tx_usecs_high = 1;
+       req->tx_usecs_high = tx_usecs_high;
+}
+static inline void
+ethtool_coalesce_set_req_set_tx_max_frames_high(struct ethtool_coalesce_set_req *req,
+                                               __u32 tx_max_frames_high)
+{
+       req->_present.tx_max_frames_high = 1;
+       req->tx_max_frames_high = tx_max_frames_high;
+}
+static inline void
+ethtool_coalesce_set_req_set_rate_sample_interval(struct ethtool_coalesce_set_req *req,
+                                                 __u32 rate_sample_interval)
+{
+       req->_present.rate_sample_interval = 1;
+       req->rate_sample_interval = rate_sample_interval;
+}
+static inline void
+ethtool_coalesce_set_req_set_use_cqe_mode_tx(struct ethtool_coalesce_set_req *req,
+                                            __u8 use_cqe_mode_tx)
+{
+       req->_present.use_cqe_mode_tx = 1;
+       req->use_cqe_mode_tx = use_cqe_mode_tx;
+}
+static inline void
+ethtool_coalesce_set_req_set_use_cqe_mode_rx(struct ethtool_coalesce_set_req *req,
+                                            __u8 use_cqe_mode_rx)
+{
+       req->_present.use_cqe_mode_rx = 1;
+       req->use_cqe_mode_rx = use_cqe_mode_rx;
+}
+static inline void
+ethtool_coalesce_set_req_set_tx_aggr_max_bytes(struct ethtool_coalesce_set_req *req,
+                                              __u32 tx_aggr_max_bytes)
+{
+       req->_present.tx_aggr_max_bytes = 1;
+       req->tx_aggr_max_bytes = tx_aggr_max_bytes;
+}
+static inline void
+ethtool_coalesce_set_req_set_tx_aggr_max_frames(struct ethtool_coalesce_set_req *req,
+                                               __u32 tx_aggr_max_frames)
+{
+       req->_present.tx_aggr_max_frames = 1;
+       req->tx_aggr_max_frames = tx_aggr_max_frames;
+}
+static inline void
+ethtool_coalesce_set_req_set_tx_aggr_time_usecs(struct ethtool_coalesce_set_req *req,
+                                               __u32 tx_aggr_time_usecs)
+{
+       req->_present.tx_aggr_time_usecs = 1;
+       req->tx_aggr_time_usecs = tx_aggr_time_usecs;
+}
+
+/*
+ * Set coalesce params.
+ */
+int ethtool_coalesce_set(struct ynl_sock *ys,
+                        struct ethtool_coalesce_set_req *req);
+
+/* ============== ETHTOOL_MSG_PAUSE_GET ============== */
+/* ETHTOOL_MSG_PAUSE_GET - do */
+struct ethtool_pause_get_req {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_pause_get_req *ethtool_pause_get_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_pause_get_req));
+}
+void ethtool_pause_get_req_free(struct ethtool_pause_get_req *req);
+
+static inline void
+ethtool_pause_get_req_set_header_dev_index(struct ethtool_pause_get_req *req,
+                                          __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_pause_get_req_set_header_dev_name(struct ethtool_pause_get_req *req,
+                                         const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_pause_get_req_set_header_flags(struct ethtool_pause_get_req *req,
+                                      __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_pause_get_rsp {
+       struct {
+               __u32 header:1;
+               __u32 autoneg:1;
+               __u32 rx:1;
+               __u32 tx:1;
+               __u32 stats:1;
+               __u32 stats_src:1;
+       } _present;
+
+       struct ethtool_header header;
+       __u8 autoneg;
+       __u8 rx;
+       __u8 tx;
+       struct ethtool_pause_stat stats;
+       __u32 stats_src;
+};
+
+void ethtool_pause_get_rsp_free(struct ethtool_pause_get_rsp *rsp);
+
+/*
+ * Get pause params.
+ */
+struct ethtool_pause_get_rsp *
+ethtool_pause_get(struct ynl_sock *ys, struct ethtool_pause_get_req *req);
+
+/* ETHTOOL_MSG_PAUSE_GET - dump */
+struct ethtool_pause_get_req_dump {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_pause_get_req_dump *
+ethtool_pause_get_req_dump_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_pause_get_req_dump));
+}
+void ethtool_pause_get_req_dump_free(struct ethtool_pause_get_req_dump *req);
+
+static inline void
+ethtool_pause_get_req_dump_set_header_dev_index(struct ethtool_pause_get_req_dump *req,
+                                               __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_pause_get_req_dump_set_header_dev_name(struct ethtool_pause_get_req_dump *req,
+                                              const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_pause_get_req_dump_set_header_flags(struct ethtool_pause_get_req_dump *req,
+                                           __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_pause_get_list {
+       struct ethtool_pause_get_list *next;
+       struct ethtool_pause_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_pause_get_list_free(struct ethtool_pause_get_list *rsp);
+
+struct ethtool_pause_get_list *
+ethtool_pause_get_dump(struct ynl_sock *ys,
+                      struct ethtool_pause_get_req_dump *req);
+
+/* ETHTOOL_MSG_PAUSE_GET - notify */
+struct ethtool_pause_get_ntf {
+       __u16 family;
+       __u8 cmd;
+       struct ynl_ntf_base_type *next;
+       void (*free)(struct ethtool_pause_get_ntf *ntf);
+       struct ethtool_pause_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_pause_get_ntf_free(struct ethtool_pause_get_ntf *rsp);
+
+/* ============== ETHTOOL_MSG_PAUSE_SET ============== */
+/* ETHTOOL_MSG_PAUSE_SET - do */
+struct ethtool_pause_set_req {
+       struct {
+               __u32 header:1;
+               __u32 autoneg:1;
+               __u32 rx:1;
+               __u32 tx:1;
+               __u32 stats:1;
+               __u32 stats_src:1;
+       } _present;
+
+       struct ethtool_header header;
+       __u8 autoneg;
+       __u8 rx;
+       __u8 tx;
+       struct ethtool_pause_stat stats;
+       __u32 stats_src;
+};
+
+static inline struct ethtool_pause_set_req *ethtool_pause_set_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_pause_set_req));
+}
+void ethtool_pause_set_req_free(struct ethtool_pause_set_req *req);
+
+static inline void
+ethtool_pause_set_req_set_header_dev_index(struct ethtool_pause_set_req *req,
+                                          __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_pause_set_req_set_header_dev_name(struct ethtool_pause_set_req *req,
+                                         const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_pause_set_req_set_header_flags(struct ethtool_pause_set_req *req,
+                                      __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+static inline void
+ethtool_pause_set_req_set_autoneg(struct ethtool_pause_set_req *req,
+                                 __u8 autoneg)
+{
+       req->_present.autoneg = 1;
+       req->autoneg = autoneg;
+}
+static inline void
+ethtool_pause_set_req_set_rx(struct ethtool_pause_set_req *req, __u8 rx)
+{
+       req->_present.rx = 1;
+       req->rx = rx;
+}
+static inline void
+ethtool_pause_set_req_set_tx(struct ethtool_pause_set_req *req, __u8 tx)
+{
+       req->_present.tx = 1;
+       req->tx = tx;
+}
+static inline void
+ethtool_pause_set_req_set_stats_tx_frames(struct ethtool_pause_set_req *req,
+                                         __u64 tx_frames)
+{
+       req->_present.stats = 1;
+       req->stats._present.tx_frames = 1;
+       req->stats.tx_frames = tx_frames;
+}
+static inline void
+ethtool_pause_set_req_set_stats_rx_frames(struct ethtool_pause_set_req *req,
+                                         __u64 rx_frames)
+{
+       req->_present.stats = 1;
+       req->stats._present.rx_frames = 1;
+       req->stats.rx_frames = rx_frames;
+}
+static inline void
+ethtool_pause_set_req_set_stats_src(struct ethtool_pause_set_req *req,
+                                   __u32 stats_src)
+{
+       req->_present.stats_src = 1;
+       req->stats_src = stats_src;
+}
+
+/*
+ * Set pause params.
+ */
+int ethtool_pause_set(struct ynl_sock *ys, struct ethtool_pause_set_req *req);
+
+/* ============== ETHTOOL_MSG_EEE_GET ============== */
+/* ETHTOOL_MSG_EEE_GET - do */
+struct ethtool_eee_get_req {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_eee_get_req *ethtool_eee_get_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_eee_get_req));
+}
+void ethtool_eee_get_req_free(struct ethtool_eee_get_req *req);
+
+static inline void
+ethtool_eee_get_req_set_header_dev_index(struct ethtool_eee_get_req *req,
+                                        __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_eee_get_req_set_header_dev_name(struct ethtool_eee_get_req *req,
+                                       const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_eee_get_req_set_header_flags(struct ethtool_eee_get_req *req,
+                                    __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_eee_get_rsp {
+       struct {
+               __u32 header:1;
+               __u32 modes_ours:1;
+               __u32 modes_peer:1;
+               __u32 active:1;
+               __u32 enabled:1;
+               __u32 tx_lpi_enabled:1;
+               __u32 tx_lpi_timer:1;
+       } _present;
+
+       struct ethtool_header header;
+       struct ethtool_bitset modes_ours;
+       struct ethtool_bitset modes_peer;
+       __u8 active;
+       __u8 enabled;
+       __u8 tx_lpi_enabled;
+       __u32 tx_lpi_timer;
+};
+
+void ethtool_eee_get_rsp_free(struct ethtool_eee_get_rsp *rsp);
+
+/*
+ * Get eee params.
+ */
+struct ethtool_eee_get_rsp *
+ethtool_eee_get(struct ynl_sock *ys, struct ethtool_eee_get_req *req);
+
+/* ETHTOOL_MSG_EEE_GET - dump */
+struct ethtool_eee_get_req_dump {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_eee_get_req_dump *
+ethtool_eee_get_req_dump_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_eee_get_req_dump));
+}
+void ethtool_eee_get_req_dump_free(struct ethtool_eee_get_req_dump *req);
+
+static inline void
+ethtool_eee_get_req_dump_set_header_dev_index(struct ethtool_eee_get_req_dump *req,
+                                             __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_eee_get_req_dump_set_header_dev_name(struct ethtool_eee_get_req_dump *req,
+                                            const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_eee_get_req_dump_set_header_flags(struct ethtool_eee_get_req_dump *req,
+                                         __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_eee_get_list {
+       struct ethtool_eee_get_list *next;
+       struct ethtool_eee_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_eee_get_list_free(struct ethtool_eee_get_list *rsp);
+
+struct ethtool_eee_get_list *
+ethtool_eee_get_dump(struct ynl_sock *ys, struct ethtool_eee_get_req_dump *req);
+
+/* ETHTOOL_MSG_EEE_GET - notify */
+struct ethtool_eee_get_ntf {
+       __u16 family;
+       __u8 cmd;
+       struct ynl_ntf_base_type *next;
+       void (*free)(struct ethtool_eee_get_ntf *ntf);
+       struct ethtool_eee_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_eee_get_ntf_free(struct ethtool_eee_get_ntf *rsp);
+
+/* ============== ETHTOOL_MSG_EEE_SET ============== */
+/* ETHTOOL_MSG_EEE_SET - do */
+struct ethtool_eee_set_req {
+       struct {
+               __u32 header:1;
+               __u32 modes_ours:1;
+               __u32 modes_peer:1;
+               __u32 active:1;
+               __u32 enabled:1;
+               __u32 tx_lpi_enabled:1;
+               __u32 tx_lpi_timer:1;
+       } _present;
+
+       struct ethtool_header header;
+       struct ethtool_bitset modes_ours;
+       struct ethtool_bitset modes_peer;
+       __u8 active;
+       __u8 enabled;
+       __u8 tx_lpi_enabled;
+       __u32 tx_lpi_timer;
+};
+
+static inline struct ethtool_eee_set_req *ethtool_eee_set_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_eee_set_req));
+}
+void ethtool_eee_set_req_free(struct ethtool_eee_set_req *req);
+
+static inline void
+ethtool_eee_set_req_set_header_dev_index(struct ethtool_eee_set_req *req,
+                                        __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_eee_set_req_set_header_dev_name(struct ethtool_eee_set_req *req,
+                                       const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_eee_set_req_set_header_flags(struct ethtool_eee_set_req *req,
+                                    __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+static inline void
+ethtool_eee_set_req_set_modes_ours_nomask(struct ethtool_eee_set_req *req)
+{
+       req->_present.modes_ours = 1;
+       req->modes_ours._present.nomask = 1;
+}
+static inline void
+ethtool_eee_set_req_set_modes_ours_size(struct ethtool_eee_set_req *req,
+                                       __u32 size)
+{
+       req->_present.modes_ours = 1;
+       req->modes_ours._present.size = 1;
+       req->modes_ours.size = size;
+}
+static inline void
+__ethtool_eee_set_req_set_modes_ours_bits_bit(struct ethtool_eee_set_req *req,
+                                             struct ethtool_bitset_bit *bit,
+                                             unsigned int n_bit)
+{
+       free(req->modes_ours.bits.bit);
+       req->modes_ours.bits.bit = bit;
+       req->modes_ours.bits.n_bit = n_bit;
+}
+static inline void
+ethtool_eee_set_req_set_modes_peer_nomask(struct ethtool_eee_set_req *req)
+{
+       req->_present.modes_peer = 1;
+       req->modes_peer._present.nomask = 1;
+}
+static inline void
+ethtool_eee_set_req_set_modes_peer_size(struct ethtool_eee_set_req *req,
+                                       __u32 size)
+{
+       req->_present.modes_peer = 1;
+       req->modes_peer._present.size = 1;
+       req->modes_peer.size = size;
+}
+static inline void
+__ethtool_eee_set_req_set_modes_peer_bits_bit(struct ethtool_eee_set_req *req,
+                                             struct ethtool_bitset_bit *bit,
+                                             unsigned int n_bit)
+{
+       free(req->modes_peer.bits.bit);
+       req->modes_peer.bits.bit = bit;
+       req->modes_peer.bits.n_bit = n_bit;
+}
+static inline void
+ethtool_eee_set_req_set_active(struct ethtool_eee_set_req *req, __u8 active)
+{
+       req->_present.active = 1;
+       req->active = active;
+}
+static inline void
+ethtool_eee_set_req_set_enabled(struct ethtool_eee_set_req *req, __u8 enabled)
+{
+       req->_present.enabled = 1;
+       req->enabled = enabled;
+}
+static inline void
+ethtool_eee_set_req_set_tx_lpi_enabled(struct ethtool_eee_set_req *req,
+                                      __u8 tx_lpi_enabled)
+{
+       req->_present.tx_lpi_enabled = 1;
+       req->tx_lpi_enabled = tx_lpi_enabled;
+}
+static inline void
+ethtool_eee_set_req_set_tx_lpi_timer(struct ethtool_eee_set_req *req,
+                                    __u32 tx_lpi_timer)
+{
+       req->_present.tx_lpi_timer = 1;
+       req->tx_lpi_timer = tx_lpi_timer;
+}
+
+/*
+ * Set eee params.
+ */
+int ethtool_eee_set(struct ynl_sock *ys, struct ethtool_eee_set_req *req);
+
+/* ============== ETHTOOL_MSG_TSINFO_GET ============== */
+/* ETHTOOL_MSG_TSINFO_GET - do */
+struct ethtool_tsinfo_get_req {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_tsinfo_get_req *ethtool_tsinfo_get_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_tsinfo_get_req));
+}
+void ethtool_tsinfo_get_req_free(struct ethtool_tsinfo_get_req *req);
+
+static inline void
+ethtool_tsinfo_get_req_set_header_dev_index(struct ethtool_tsinfo_get_req *req,
+                                           __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_tsinfo_get_req_set_header_dev_name(struct ethtool_tsinfo_get_req *req,
+                                          const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_tsinfo_get_req_set_header_flags(struct ethtool_tsinfo_get_req *req,
+                                       __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_tsinfo_get_rsp {
+       struct {
+               __u32 header:1;
+               __u32 timestamping:1;
+               __u32 tx_types:1;
+               __u32 rx_filters:1;
+               __u32 phc_index:1;
+       } _present;
+
+       struct ethtool_header header;
+       struct ethtool_bitset timestamping;
+       struct ethtool_bitset tx_types;
+       struct ethtool_bitset rx_filters;
+       __u32 phc_index;
+};
+
+void ethtool_tsinfo_get_rsp_free(struct ethtool_tsinfo_get_rsp *rsp);
+
+/*
+ * Get tsinfo params.
+ */
+struct ethtool_tsinfo_get_rsp *
+ethtool_tsinfo_get(struct ynl_sock *ys, struct ethtool_tsinfo_get_req *req);
+
+/* ETHTOOL_MSG_TSINFO_GET - dump */
+struct ethtool_tsinfo_get_req_dump {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_tsinfo_get_req_dump *
+ethtool_tsinfo_get_req_dump_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_tsinfo_get_req_dump));
+}
+void ethtool_tsinfo_get_req_dump_free(struct ethtool_tsinfo_get_req_dump *req);
+
+static inline void
+ethtool_tsinfo_get_req_dump_set_header_dev_index(struct ethtool_tsinfo_get_req_dump *req,
+                                                __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_tsinfo_get_req_dump_set_header_dev_name(struct ethtool_tsinfo_get_req_dump *req,
+                                               const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_tsinfo_get_req_dump_set_header_flags(struct ethtool_tsinfo_get_req_dump *req,
+                                            __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_tsinfo_get_list {
+       struct ethtool_tsinfo_get_list *next;
+       struct ethtool_tsinfo_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_tsinfo_get_list_free(struct ethtool_tsinfo_get_list *rsp);
+
+struct ethtool_tsinfo_get_list *
+ethtool_tsinfo_get_dump(struct ynl_sock *ys,
+                       struct ethtool_tsinfo_get_req_dump *req);
+
+/* ============== ETHTOOL_MSG_CABLE_TEST_ACT ============== */
+/* ETHTOOL_MSG_CABLE_TEST_ACT - do */
+struct ethtool_cable_test_act_req {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_cable_test_act_req *
+ethtool_cable_test_act_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_cable_test_act_req));
+}
+void ethtool_cable_test_act_req_free(struct ethtool_cable_test_act_req *req);
+
+static inline void
+ethtool_cable_test_act_req_set_header_dev_index(struct ethtool_cable_test_act_req *req,
+                                               __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_cable_test_act_req_set_header_dev_name(struct ethtool_cable_test_act_req *req,
+                                              const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_cable_test_act_req_set_header_flags(struct ethtool_cable_test_act_req *req,
+                                           __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+/*
+ * Cable test.
+ */
+int ethtool_cable_test_act(struct ynl_sock *ys,
+                          struct ethtool_cable_test_act_req *req);
+
+/* ============== ETHTOOL_MSG_CABLE_TEST_TDR_ACT ============== */
+/* ETHTOOL_MSG_CABLE_TEST_TDR_ACT - do */
+struct ethtool_cable_test_tdr_act_req {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_cable_test_tdr_act_req *
+ethtool_cable_test_tdr_act_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_cable_test_tdr_act_req));
+}
+void
+ethtool_cable_test_tdr_act_req_free(struct ethtool_cable_test_tdr_act_req *req);
+
+static inline void
+ethtool_cable_test_tdr_act_req_set_header_dev_index(struct ethtool_cable_test_tdr_act_req *req,
+                                                   __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_cable_test_tdr_act_req_set_header_dev_name(struct ethtool_cable_test_tdr_act_req *req,
+                                                  const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_cable_test_tdr_act_req_set_header_flags(struct ethtool_cable_test_tdr_act_req *req,
+                                               __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+/*
+ * Cable test TDR.
+ */
+int ethtool_cable_test_tdr_act(struct ynl_sock *ys,
+                              struct ethtool_cable_test_tdr_act_req *req);
+
+/* ============== ETHTOOL_MSG_TUNNEL_INFO_GET ============== */
+/* ETHTOOL_MSG_TUNNEL_INFO_GET - do */
+struct ethtool_tunnel_info_get_req {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_tunnel_info_get_req *
+ethtool_tunnel_info_get_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_tunnel_info_get_req));
+}
+void ethtool_tunnel_info_get_req_free(struct ethtool_tunnel_info_get_req *req);
+
+static inline void
+ethtool_tunnel_info_get_req_set_header_dev_index(struct ethtool_tunnel_info_get_req *req,
+                                                __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_tunnel_info_get_req_set_header_dev_name(struct ethtool_tunnel_info_get_req *req,
+                                               const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_tunnel_info_get_req_set_header_flags(struct ethtool_tunnel_info_get_req *req,
+                                            __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_tunnel_info_get_rsp {
+       struct {
+               __u32 header:1;
+               __u32 udp_ports:1;
+       } _present;
+
+       struct ethtool_header header;
+       struct ethtool_tunnel_udp udp_ports;
+};
+
+void ethtool_tunnel_info_get_rsp_free(struct ethtool_tunnel_info_get_rsp *rsp);
+
+/*
+ * Get tsinfo params.
+ */
+struct ethtool_tunnel_info_get_rsp *
+ethtool_tunnel_info_get(struct ynl_sock *ys,
+                       struct ethtool_tunnel_info_get_req *req);
+
+/* ETHTOOL_MSG_TUNNEL_INFO_GET - dump */
+struct ethtool_tunnel_info_get_req_dump {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_tunnel_info_get_req_dump *
+ethtool_tunnel_info_get_req_dump_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_tunnel_info_get_req_dump));
+}
+void
+ethtool_tunnel_info_get_req_dump_free(struct ethtool_tunnel_info_get_req_dump *req);
+
+static inline void
+ethtool_tunnel_info_get_req_dump_set_header_dev_index(struct ethtool_tunnel_info_get_req_dump *req,
+                                                     __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_tunnel_info_get_req_dump_set_header_dev_name(struct ethtool_tunnel_info_get_req_dump *req,
+                                                    const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_tunnel_info_get_req_dump_set_header_flags(struct ethtool_tunnel_info_get_req_dump *req,
+                                                 __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_tunnel_info_get_list {
+       struct ethtool_tunnel_info_get_list *next;
+       struct ethtool_tunnel_info_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void
+ethtool_tunnel_info_get_list_free(struct ethtool_tunnel_info_get_list *rsp);
+
+struct ethtool_tunnel_info_get_list *
+ethtool_tunnel_info_get_dump(struct ynl_sock *ys,
+                            struct ethtool_tunnel_info_get_req_dump *req);
+
+/* ============== ETHTOOL_MSG_FEC_GET ============== */
+/* ETHTOOL_MSG_FEC_GET - do */
+struct ethtool_fec_get_req {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_fec_get_req *ethtool_fec_get_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_fec_get_req));
+}
+void ethtool_fec_get_req_free(struct ethtool_fec_get_req *req);
+
+static inline void
+ethtool_fec_get_req_set_header_dev_index(struct ethtool_fec_get_req *req,
+                                        __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_fec_get_req_set_header_dev_name(struct ethtool_fec_get_req *req,
+                                       const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_fec_get_req_set_header_flags(struct ethtool_fec_get_req *req,
+                                    __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_fec_get_rsp {
+       struct {
+               __u32 header:1;
+               __u32 modes:1;
+               __u32 auto_:1;
+               __u32 active:1;
+               __u32 stats:1;
+       } _present;
+
+       struct ethtool_header header;
+       struct ethtool_bitset modes;
+       __u8 auto_;
+       __u32 active;
+       struct ethtool_fec_stat stats;
+};
+
+void ethtool_fec_get_rsp_free(struct ethtool_fec_get_rsp *rsp);
+
+/*
+ * Get FEC params.
+ */
+struct ethtool_fec_get_rsp *
+ethtool_fec_get(struct ynl_sock *ys, struct ethtool_fec_get_req *req);
+
+/* ETHTOOL_MSG_FEC_GET - dump */
+struct ethtool_fec_get_req_dump {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_fec_get_req_dump *
+ethtool_fec_get_req_dump_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_fec_get_req_dump));
+}
+void ethtool_fec_get_req_dump_free(struct ethtool_fec_get_req_dump *req);
+
+static inline void
+ethtool_fec_get_req_dump_set_header_dev_index(struct ethtool_fec_get_req_dump *req,
+                                             __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_fec_get_req_dump_set_header_dev_name(struct ethtool_fec_get_req_dump *req,
+                                            const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_fec_get_req_dump_set_header_flags(struct ethtool_fec_get_req_dump *req,
+                                         __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_fec_get_list {
+       struct ethtool_fec_get_list *next;
+       struct ethtool_fec_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_fec_get_list_free(struct ethtool_fec_get_list *rsp);
+
+struct ethtool_fec_get_list *
+ethtool_fec_get_dump(struct ynl_sock *ys, struct ethtool_fec_get_req_dump *req);
+
+/* ETHTOOL_MSG_FEC_GET - notify */
+struct ethtool_fec_get_ntf {
+       __u16 family;
+       __u8 cmd;
+       struct ynl_ntf_base_type *next;
+       void (*free)(struct ethtool_fec_get_ntf *ntf);
+       struct ethtool_fec_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_fec_get_ntf_free(struct ethtool_fec_get_ntf *rsp);
+
+/* ============== ETHTOOL_MSG_FEC_SET ============== */
+/* ETHTOOL_MSG_FEC_SET - do */
+struct ethtool_fec_set_req {
+       struct {
+               __u32 header:1;
+               __u32 modes:1;
+               __u32 auto_:1;
+               __u32 active:1;
+               __u32 stats:1;
+       } _present;
+
+       struct ethtool_header header;
+       struct ethtool_bitset modes;
+       __u8 auto_;
+       __u32 active;
+       struct ethtool_fec_stat stats;
+};
+
+static inline struct ethtool_fec_set_req *ethtool_fec_set_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_fec_set_req));
+}
+void ethtool_fec_set_req_free(struct ethtool_fec_set_req *req);
+
+static inline void
+ethtool_fec_set_req_set_header_dev_index(struct ethtool_fec_set_req *req,
+                                        __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_fec_set_req_set_header_dev_name(struct ethtool_fec_set_req *req,
+                                       const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_fec_set_req_set_header_flags(struct ethtool_fec_set_req *req,
+                                    __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+static inline void
+ethtool_fec_set_req_set_modes_nomask(struct ethtool_fec_set_req *req)
+{
+       req->_present.modes = 1;
+       req->modes._present.nomask = 1;
+}
+static inline void
+ethtool_fec_set_req_set_modes_size(struct ethtool_fec_set_req *req, __u32 size)
+{
+       req->_present.modes = 1;
+       req->modes._present.size = 1;
+       req->modes.size = size;
+}
+static inline void
+__ethtool_fec_set_req_set_modes_bits_bit(struct ethtool_fec_set_req *req,
+                                        struct ethtool_bitset_bit *bit,
+                                        unsigned int n_bit)
+{
+       free(req->modes.bits.bit);
+       req->modes.bits.bit = bit;
+       req->modes.bits.n_bit = n_bit;
+}
+static inline void
+ethtool_fec_set_req_set_auto_(struct ethtool_fec_set_req *req, __u8 auto_)
+{
+       req->_present.auto_ = 1;
+       req->auto_ = auto_;
+}
+static inline void
+ethtool_fec_set_req_set_active(struct ethtool_fec_set_req *req, __u32 active)
+{
+       req->_present.active = 1;
+       req->active = active;
+}
+static inline void
+ethtool_fec_set_req_set_stats_corrected(struct ethtool_fec_set_req *req,
+                                       const void *corrected, size_t len)
+{
+       free(req->stats.corrected);
+       req->stats.corrected = malloc(req->stats._present.corrected_len);
+       memcpy(req->stats.corrected, corrected, req->stats._present.corrected_len);
+}
+static inline void
+ethtool_fec_set_req_set_stats_uncorr(struct ethtool_fec_set_req *req,
+                                    const void *uncorr, size_t len)
+{
+       free(req->stats.uncorr);
+       req->stats.uncorr = malloc(req->stats._present.uncorr_len);
+       memcpy(req->stats.uncorr, uncorr, req->stats._present.uncorr_len);
+}
+static inline void
+ethtool_fec_set_req_set_stats_corr_bits(struct ethtool_fec_set_req *req,
+                                       const void *corr_bits, size_t len)
+{
+       free(req->stats.corr_bits);
+       req->stats.corr_bits = malloc(req->stats._present.corr_bits_len);
+       memcpy(req->stats.corr_bits, corr_bits, req->stats._present.corr_bits_len);
+}
+
+/*
+ * Set FEC params.
+ */
+int ethtool_fec_set(struct ynl_sock *ys, struct ethtool_fec_set_req *req);
+
+/* ============== ETHTOOL_MSG_MODULE_EEPROM_GET ============== */
+/* ETHTOOL_MSG_MODULE_EEPROM_GET - do */
+struct ethtool_module_eeprom_get_req {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_module_eeprom_get_req *
+ethtool_module_eeprom_get_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_module_eeprom_get_req));
+}
+void
+ethtool_module_eeprom_get_req_free(struct ethtool_module_eeprom_get_req *req);
+
+static inline void
+ethtool_module_eeprom_get_req_set_header_dev_index(struct ethtool_module_eeprom_get_req *req,
+                                                  __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_module_eeprom_get_req_set_header_dev_name(struct ethtool_module_eeprom_get_req *req,
+                                                 const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_module_eeprom_get_req_set_header_flags(struct ethtool_module_eeprom_get_req *req,
+                                              __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_module_eeprom_get_rsp {
+       struct {
+               __u32 header:1;
+               __u32 offset:1;
+               __u32 length:1;
+               __u32 page:1;
+               __u32 bank:1;
+               __u32 i2c_address:1;
+               __u32 data_len;
+       } _present;
+
+       struct ethtool_header header;
+       __u32 offset;
+       __u32 length;
+       __u8 page;
+       __u8 bank;
+       __u8 i2c_address;
+       void *data;
+};
+
+void
+ethtool_module_eeprom_get_rsp_free(struct ethtool_module_eeprom_get_rsp *rsp);
+
+/*
+ * Get module EEPROM params.
+ */
+struct ethtool_module_eeprom_get_rsp *
+ethtool_module_eeprom_get(struct ynl_sock *ys,
+                         struct ethtool_module_eeprom_get_req *req);
+
+/* ETHTOOL_MSG_MODULE_EEPROM_GET - dump */
+struct ethtool_module_eeprom_get_req_dump {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_module_eeprom_get_req_dump *
+ethtool_module_eeprom_get_req_dump_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_module_eeprom_get_req_dump));
+}
+void
+ethtool_module_eeprom_get_req_dump_free(struct ethtool_module_eeprom_get_req_dump *req);
+
+static inline void
+ethtool_module_eeprom_get_req_dump_set_header_dev_index(struct ethtool_module_eeprom_get_req_dump *req,
+                                                       __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_module_eeprom_get_req_dump_set_header_dev_name(struct ethtool_module_eeprom_get_req_dump *req,
+                                                      const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_module_eeprom_get_req_dump_set_header_flags(struct ethtool_module_eeprom_get_req_dump *req,
+                                                   __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_module_eeprom_get_list {
+       struct ethtool_module_eeprom_get_list *next;
+       struct ethtool_module_eeprom_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void
+ethtool_module_eeprom_get_list_free(struct ethtool_module_eeprom_get_list *rsp);
+
+struct ethtool_module_eeprom_get_list *
+ethtool_module_eeprom_get_dump(struct ynl_sock *ys,
+                              struct ethtool_module_eeprom_get_req_dump *req);
+
+/* ============== ETHTOOL_MSG_PHC_VCLOCKS_GET ============== */
+/* ETHTOOL_MSG_PHC_VCLOCKS_GET - do */
+struct ethtool_phc_vclocks_get_req {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_phc_vclocks_get_req *
+ethtool_phc_vclocks_get_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_phc_vclocks_get_req));
+}
+void ethtool_phc_vclocks_get_req_free(struct ethtool_phc_vclocks_get_req *req);
+
+static inline void
+ethtool_phc_vclocks_get_req_set_header_dev_index(struct ethtool_phc_vclocks_get_req *req,
+                                                __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_phc_vclocks_get_req_set_header_dev_name(struct ethtool_phc_vclocks_get_req *req,
+                                               const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_phc_vclocks_get_req_set_header_flags(struct ethtool_phc_vclocks_get_req *req,
+                                            __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_phc_vclocks_get_rsp {
+       struct {
+               __u32 header:1;
+               __u32 num:1;
+       } _present;
+
+       struct ethtool_header header;
+       __u32 num;
+};
+
+void ethtool_phc_vclocks_get_rsp_free(struct ethtool_phc_vclocks_get_rsp *rsp);
+
+/*
+ * Get PHC VCLOCKs.
+ */
+struct ethtool_phc_vclocks_get_rsp *
+ethtool_phc_vclocks_get(struct ynl_sock *ys,
+                       struct ethtool_phc_vclocks_get_req *req);
+
+/* ETHTOOL_MSG_PHC_VCLOCKS_GET - dump */
+struct ethtool_phc_vclocks_get_req_dump {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_phc_vclocks_get_req_dump *
+ethtool_phc_vclocks_get_req_dump_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_phc_vclocks_get_req_dump));
+}
+void
+ethtool_phc_vclocks_get_req_dump_free(struct ethtool_phc_vclocks_get_req_dump *req);
+
+static inline void
+ethtool_phc_vclocks_get_req_dump_set_header_dev_index(struct ethtool_phc_vclocks_get_req_dump *req,
+                                                     __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_phc_vclocks_get_req_dump_set_header_dev_name(struct ethtool_phc_vclocks_get_req_dump *req,
+                                                    const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_phc_vclocks_get_req_dump_set_header_flags(struct ethtool_phc_vclocks_get_req_dump *req,
+                                                 __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_phc_vclocks_get_list {
+       struct ethtool_phc_vclocks_get_list *next;
+       struct ethtool_phc_vclocks_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void
+ethtool_phc_vclocks_get_list_free(struct ethtool_phc_vclocks_get_list *rsp);
+
+struct ethtool_phc_vclocks_get_list *
+ethtool_phc_vclocks_get_dump(struct ynl_sock *ys,
+                            struct ethtool_phc_vclocks_get_req_dump *req);
+
+/* ============== ETHTOOL_MSG_MODULE_GET ============== */
+/* ETHTOOL_MSG_MODULE_GET - do */
+struct ethtool_module_get_req {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_module_get_req *ethtool_module_get_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_module_get_req));
+}
+void ethtool_module_get_req_free(struct ethtool_module_get_req *req);
+
+static inline void
+ethtool_module_get_req_set_header_dev_index(struct ethtool_module_get_req *req,
+                                           __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_module_get_req_set_header_dev_name(struct ethtool_module_get_req *req,
+                                          const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_module_get_req_set_header_flags(struct ethtool_module_get_req *req,
+                                       __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_module_get_rsp {
+       struct {
+               __u32 header:1;
+               __u32 power_mode_policy:1;
+               __u32 power_mode:1;
+       } _present;
+
+       struct ethtool_header header;
+       __u8 power_mode_policy;
+       __u8 power_mode;
+};
+
+void ethtool_module_get_rsp_free(struct ethtool_module_get_rsp *rsp);
+
+/*
+ * Get module params.
+ */
+struct ethtool_module_get_rsp *
+ethtool_module_get(struct ynl_sock *ys, struct ethtool_module_get_req *req);
+
+/* ETHTOOL_MSG_MODULE_GET - dump */
+struct ethtool_module_get_req_dump {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_module_get_req_dump *
+ethtool_module_get_req_dump_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_module_get_req_dump));
+}
+void ethtool_module_get_req_dump_free(struct ethtool_module_get_req_dump *req);
+
+static inline void
+ethtool_module_get_req_dump_set_header_dev_index(struct ethtool_module_get_req_dump *req,
+                                                __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_module_get_req_dump_set_header_dev_name(struct ethtool_module_get_req_dump *req,
+                                               const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_module_get_req_dump_set_header_flags(struct ethtool_module_get_req_dump *req,
+                                            __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_module_get_list {
+       struct ethtool_module_get_list *next;
+       struct ethtool_module_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_module_get_list_free(struct ethtool_module_get_list *rsp);
+
+struct ethtool_module_get_list *
+ethtool_module_get_dump(struct ynl_sock *ys,
+                       struct ethtool_module_get_req_dump *req);
+
+/* ETHTOOL_MSG_MODULE_GET - notify */
+struct ethtool_module_get_ntf {
+       __u16 family;
+       __u8 cmd;
+       struct ynl_ntf_base_type *next;
+       void (*free)(struct ethtool_module_get_ntf *ntf);
+       struct ethtool_module_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_module_get_ntf_free(struct ethtool_module_get_ntf *rsp);
+
+/* ============== ETHTOOL_MSG_MODULE_SET ============== */
+/* ETHTOOL_MSG_MODULE_SET - do */
+struct ethtool_module_set_req {
+       struct {
+               __u32 header:1;
+               __u32 power_mode_policy:1;
+               __u32 power_mode:1;
+       } _present;
+
+       struct ethtool_header header;
+       __u8 power_mode_policy;
+       __u8 power_mode;
+};
+
+static inline struct ethtool_module_set_req *ethtool_module_set_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_module_set_req));
+}
+void ethtool_module_set_req_free(struct ethtool_module_set_req *req);
+
+static inline void
+ethtool_module_set_req_set_header_dev_index(struct ethtool_module_set_req *req,
+                                           __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_module_set_req_set_header_dev_name(struct ethtool_module_set_req *req,
+                                          const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_module_set_req_set_header_flags(struct ethtool_module_set_req *req,
+                                       __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+static inline void
+ethtool_module_set_req_set_power_mode_policy(struct ethtool_module_set_req *req,
+                                            __u8 power_mode_policy)
+{
+       req->_present.power_mode_policy = 1;
+       req->power_mode_policy = power_mode_policy;
+}
+static inline void
+ethtool_module_set_req_set_power_mode(struct ethtool_module_set_req *req,
+                                     __u8 power_mode)
+{
+       req->_present.power_mode = 1;
+       req->power_mode = power_mode;
+}
+
+/*
+ * Set module params.
+ */
+int ethtool_module_set(struct ynl_sock *ys, struct ethtool_module_set_req *req);
+
+/* ============== ETHTOOL_MSG_PSE_GET ============== */
+/* ETHTOOL_MSG_PSE_GET - do */
+struct ethtool_pse_get_req {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_pse_get_req *ethtool_pse_get_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_pse_get_req));
+}
+void ethtool_pse_get_req_free(struct ethtool_pse_get_req *req);
+
+static inline void
+ethtool_pse_get_req_set_header_dev_index(struct ethtool_pse_get_req *req,
+                                        __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_pse_get_req_set_header_dev_name(struct ethtool_pse_get_req *req,
+                                       const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_pse_get_req_set_header_flags(struct ethtool_pse_get_req *req,
+                                    __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_pse_get_rsp {
+       struct {
+               __u32 header:1;
+               __u32 admin_state:1;
+               __u32 admin_control:1;
+               __u32 pw_d_status:1;
+       } _present;
+
+       struct ethtool_header header;
+       __u32 admin_state;
+       __u32 admin_control;
+       __u32 pw_d_status;
+};
+
+void ethtool_pse_get_rsp_free(struct ethtool_pse_get_rsp *rsp);
+
+/*
+ * Get Power Sourcing Equipment params.
+ */
+struct ethtool_pse_get_rsp *
+ethtool_pse_get(struct ynl_sock *ys, struct ethtool_pse_get_req *req);
+
+/* ETHTOOL_MSG_PSE_GET - dump */
+struct ethtool_pse_get_req_dump {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_pse_get_req_dump *
+ethtool_pse_get_req_dump_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_pse_get_req_dump));
+}
+void ethtool_pse_get_req_dump_free(struct ethtool_pse_get_req_dump *req);
+
+static inline void
+ethtool_pse_get_req_dump_set_header_dev_index(struct ethtool_pse_get_req_dump *req,
+                                             __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_pse_get_req_dump_set_header_dev_name(struct ethtool_pse_get_req_dump *req,
+                                            const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_pse_get_req_dump_set_header_flags(struct ethtool_pse_get_req_dump *req,
+                                         __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_pse_get_list {
+       struct ethtool_pse_get_list *next;
+       struct ethtool_pse_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_pse_get_list_free(struct ethtool_pse_get_list *rsp);
+
+struct ethtool_pse_get_list *
+ethtool_pse_get_dump(struct ynl_sock *ys, struct ethtool_pse_get_req_dump *req);
+
+/* ============== ETHTOOL_MSG_PSE_SET ============== */
+/* ETHTOOL_MSG_PSE_SET - do */
+struct ethtool_pse_set_req {
+       struct {
+               __u32 header:1;
+               __u32 admin_state:1;
+               __u32 admin_control:1;
+               __u32 pw_d_status:1;
+       } _present;
+
+       struct ethtool_header header;
+       __u32 admin_state;
+       __u32 admin_control;
+       __u32 pw_d_status;
+};
+
+static inline struct ethtool_pse_set_req *ethtool_pse_set_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_pse_set_req));
+}
+void ethtool_pse_set_req_free(struct ethtool_pse_set_req *req);
+
+static inline void
+ethtool_pse_set_req_set_header_dev_index(struct ethtool_pse_set_req *req,
+                                        __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_pse_set_req_set_header_dev_name(struct ethtool_pse_set_req *req,
+                                       const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_pse_set_req_set_header_flags(struct ethtool_pse_set_req *req,
+                                    __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+static inline void
+ethtool_pse_set_req_set_admin_state(struct ethtool_pse_set_req *req,
+                                   __u32 admin_state)
+{
+       req->_present.admin_state = 1;
+       req->admin_state = admin_state;
+}
+static inline void
+ethtool_pse_set_req_set_admin_control(struct ethtool_pse_set_req *req,
+                                     __u32 admin_control)
+{
+       req->_present.admin_control = 1;
+       req->admin_control = admin_control;
+}
+static inline void
+ethtool_pse_set_req_set_pw_d_status(struct ethtool_pse_set_req *req,
+                                   __u32 pw_d_status)
+{
+       req->_present.pw_d_status = 1;
+       req->pw_d_status = pw_d_status;
+}
+
+/*
+ * Set Power Sourcing Equipment params.
+ */
+int ethtool_pse_set(struct ynl_sock *ys, struct ethtool_pse_set_req *req);
+
+/* ============== ETHTOOL_MSG_RSS_GET ============== */
+/* ETHTOOL_MSG_RSS_GET - do */
+struct ethtool_rss_get_req {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_rss_get_req *ethtool_rss_get_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_rss_get_req));
+}
+void ethtool_rss_get_req_free(struct ethtool_rss_get_req *req);
+
+static inline void
+ethtool_rss_get_req_set_header_dev_index(struct ethtool_rss_get_req *req,
+                                        __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_rss_get_req_set_header_dev_name(struct ethtool_rss_get_req *req,
+                                       const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_rss_get_req_set_header_flags(struct ethtool_rss_get_req *req,
+                                    __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_rss_get_rsp {
+       struct {
+               __u32 header:1;
+               __u32 context:1;
+               __u32 hfunc:1;
+               __u32 indir_len;
+               __u32 hkey_len;
+       } _present;
+
+       struct ethtool_header header;
+       __u32 context;
+       __u32 hfunc;
+       void *indir;
+       void *hkey;
+};
+
+void ethtool_rss_get_rsp_free(struct ethtool_rss_get_rsp *rsp);
+
+/*
+ * Get RSS params.
+ */
+struct ethtool_rss_get_rsp *
+ethtool_rss_get(struct ynl_sock *ys, struct ethtool_rss_get_req *req);
+
+/* ETHTOOL_MSG_RSS_GET - dump */
+struct ethtool_rss_get_req_dump {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_rss_get_req_dump *
+ethtool_rss_get_req_dump_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_rss_get_req_dump));
+}
+void ethtool_rss_get_req_dump_free(struct ethtool_rss_get_req_dump *req);
+
+static inline void
+ethtool_rss_get_req_dump_set_header_dev_index(struct ethtool_rss_get_req_dump *req,
+                                             __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_rss_get_req_dump_set_header_dev_name(struct ethtool_rss_get_req_dump *req,
+                                            const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_rss_get_req_dump_set_header_flags(struct ethtool_rss_get_req_dump *req,
+                                         __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_rss_get_list {
+       struct ethtool_rss_get_list *next;
+       struct ethtool_rss_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_rss_get_list_free(struct ethtool_rss_get_list *rsp);
+
+struct ethtool_rss_get_list *
+ethtool_rss_get_dump(struct ynl_sock *ys, struct ethtool_rss_get_req_dump *req);
+
+/* ============== ETHTOOL_MSG_PLCA_GET_CFG ============== */
+/* ETHTOOL_MSG_PLCA_GET_CFG - do */
+struct ethtool_plca_get_cfg_req {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_plca_get_cfg_req *
+ethtool_plca_get_cfg_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_plca_get_cfg_req));
+}
+void ethtool_plca_get_cfg_req_free(struct ethtool_plca_get_cfg_req *req);
+
+static inline void
+ethtool_plca_get_cfg_req_set_header_dev_index(struct ethtool_plca_get_cfg_req *req,
+                                             __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_plca_get_cfg_req_set_header_dev_name(struct ethtool_plca_get_cfg_req *req,
+                                            const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_plca_get_cfg_req_set_header_flags(struct ethtool_plca_get_cfg_req *req,
+                                         __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_plca_get_cfg_rsp {
+       struct {
+               __u32 header:1;
+               __u32 version:1;
+               __u32 enabled:1;
+               __u32 status:1;
+               __u32 node_cnt:1;
+               __u32 node_id:1;
+               __u32 to_tmr:1;
+               __u32 burst_cnt:1;
+               __u32 burst_tmr:1;
+       } _present;
+
+       struct ethtool_header header;
+       __u16 version;
+       __u8 enabled;
+       __u8 status;
+       __u32 node_cnt;
+       __u32 node_id;
+       __u32 to_tmr;
+       __u32 burst_cnt;
+       __u32 burst_tmr;
+};
+
+void ethtool_plca_get_cfg_rsp_free(struct ethtool_plca_get_cfg_rsp *rsp);
+
+/*
+ * Get PLCA params.
+ */
+struct ethtool_plca_get_cfg_rsp *
+ethtool_plca_get_cfg(struct ynl_sock *ys, struct ethtool_plca_get_cfg_req *req);
+
+/* ETHTOOL_MSG_PLCA_GET_CFG - dump */
+struct ethtool_plca_get_cfg_req_dump {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_plca_get_cfg_req_dump *
+ethtool_plca_get_cfg_req_dump_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_plca_get_cfg_req_dump));
+}
+void
+ethtool_plca_get_cfg_req_dump_free(struct ethtool_plca_get_cfg_req_dump *req);
+
+static inline void
+ethtool_plca_get_cfg_req_dump_set_header_dev_index(struct ethtool_plca_get_cfg_req_dump *req,
+                                                  __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_plca_get_cfg_req_dump_set_header_dev_name(struct ethtool_plca_get_cfg_req_dump *req,
+                                                 const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_plca_get_cfg_req_dump_set_header_flags(struct ethtool_plca_get_cfg_req_dump *req,
+                                              __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_plca_get_cfg_list {
+       struct ethtool_plca_get_cfg_list *next;
+       struct ethtool_plca_get_cfg_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_plca_get_cfg_list_free(struct ethtool_plca_get_cfg_list *rsp);
+
+struct ethtool_plca_get_cfg_list *
+ethtool_plca_get_cfg_dump(struct ynl_sock *ys,
+                         struct ethtool_plca_get_cfg_req_dump *req);
+
+/* ETHTOOL_MSG_PLCA_GET_CFG - notify */
+struct ethtool_plca_get_cfg_ntf {
+       __u16 family;
+       __u8 cmd;
+       struct ynl_ntf_base_type *next;
+       void (*free)(struct ethtool_plca_get_cfg_ntf *ntf);
+       struct ethtool_plca_get_cfg_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_plca_get_cfg_ntf_free(struct ethtool_plca_get_cfg_ntf *rsp);
+
+/* ============== ETHTOOL_MSG_PLCA_SET_CFG ============== */
+/* ETHTOOL_MSG_PLCA_SET_CFG - do */
+struct ethtool_plca_set_cfg_req {
+       struct {
+               __u32 header:1;
+               __u32 version:1;
+               __u32 enabled:1;
+               __u32 status:1;
+               __u32 node_cnt:1;
+               __u32 node_id:1;
+               __u32 to_tmr:1;
+               __u32 burst_cnt:1;
+               __u32 burst_tmr:1;
+       } _present;
+
+       struct ethtool_header header;
+       __u16 version;
+       __u8 enabled;
+       __u8 status;
+       __u32 node_cnt;
+       __u32 node_id;
+       __u32 to_tmr;
+       __u32 burst_cnt;
+       __u32 burst_tmr;
+};
+
+static inline struct ethtool_plca_set_cfg_req *
+ethtool_plca_set_cfg_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_plca_set_cfg_req));
+}
+void ethtool_plca_set_cfg_req_free(struct ethtool_plca_set_cfg_req *req);
+
+static inline void
+ethtool_plca_set_cfg_req_set_header_dev_index(struct ethtool_plca_set_cfg_req *req,
+                                             __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_plca_set_cfg_req_set_header_dev_name(struct ethtool_plca_set_cfg_req *req,
+                                            const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_plca_set_cfg_req_set_header_flags(struct ethtool_plca_set_cfg_req *req,
+                                         __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+static inline void
+ethtool_plca_set_cfg_req_set_version(struct ethtool_plca_set_cfg_req *req,
+                                    __u16 version)
+{
+       req->_present.version = 1;
+       req->version = version;
+}
+static inline void
+ethtool_plca_set_cfg_req_set_enabled(struct ethtool_plca_set_cfg_req *req,
+                                    __u8 enabled)
+{
+       req->_present.enabled = 1;
+       req->enabled = enabled;
+}
+static inline void
+ethtool_plca_set_cfg_req_set_status(struct ethtool_plca_set_cfg_req *req,
+                                   __u8 status)
+{
+       req->_present.status = 1;
+       req->status = status;
+}
+static inline void
+ethtool_plca_set_cfg_req_set_node_cnt(struct ethtool_plca_set_cfg_req *req,
+                                     __u32 node_cnt)
+{
+       req->_present.node_cnt = 1;
+       req->node_cnt = node_cnt;
+}
+static inline void
+ethtool_plca_set_cfg_req_set_node_id(struct ethtool_plca_set_cfg_req *req,
+                                    __u32 node_id)
+{
+       req->_present.node_id = 1;
+       req->node_id = node_id;
+}
+static inline void
+ethtool_plca_set_cfg_req_set_to_tmr(struct ethtool_plca_set_cfg_req *req,
+                                   __u32 to_tmr)
+{
+       req->_present.to_tmr = 1;
+       req->to_tmr = to_tmr;
+}
+static inline void
+ethtool_plca_set_cfg_req_set_burst_cnt(struct ethtool_plca_set_cfg_req *req,
+                                      __u32 burst_cnt)
+{
+       req->_present.burst_cnt = 1;
+       req->burst_cnt = burst_cnt;
+}
+static inline void
+ethtool_plca_set_cfg_req_set_burst_tmr(struct ethtool_plca_set_cfg_req *req,
+                                      __u32 burst_tmr)
+{
+       req->_present.burst_tmr = 1;
+       req->burst_tmr = burst_tmr;
+}
+
+/*
+ * Set PLCA params.
+ */
+int ethtool_plca_set_cfg(struct ynl_sock *ys,
+                        struct ethtool_plca_set_cfg_req *req);
+
+/* ============== ETHTOOL_MSG_PLCA_GET_STATUS ============== */
+/* ETHTOOL_MSG_PLCA_GET_STATUS - do */
+struct ethtool_plca_get_status_req {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_plca_get_status_req *
+ethtool_plca_get_status_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_plca_get_status_req));
+}
+void ethtool_plca_get_status_req_free(struct ethtool_plca_get_status_req *req);
+
+static inline void
+ethtool_plca_get_status_req_set_header_dev_index(struct ethtool_plca_get_status_req *req,
+                                                __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_plca_get_status_req_set_header_dev_name(struct ethtool_plca_get_status_req *req,
+                                               const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_plca_get_status_req_set_header_flags(struct ethtool_plca_get_status_req *req,
+                                            __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_plca_get_status_rsp {
+       struct {
+               __u32 header:1;
+               __u32 version:1;
+               __u32 enabled:1;
+               __u32 status:1;
+               __u32 node_cnt:1;
+               __u32 node_id:1;
+               __u32 to_tmr:1;
+               __u32 burst_cnt:1;
+               __u32 burst_tmr:1;
+       } _present;
+
+       struct ethtool_header header;
+       __u16 version;
+       __u8 enabled;
+       __u8 status;
+       __u32 node_cnt;
+       __u32 node_id;
+       __u32 to_tmr;
+       __u32 burst_cnt;
+       __u32 burst_tmr;
+};
+
+void ethtool_plca_get_status_rsp_free(struct ethtool_plca_get_status_rsp *rsp);
+
+/*
+ * Get PLCA status params.
+ */
+struct ethtool_plca_get_status_rsp *
+ethtool_plca_get_status(struct ynl_sock *ys,
+                       struct ethtool_plca_get_status_req *req);
+
+/* ETHTOOL_MSG_PLCA_GET_STATUS - dump */
+struct ethtool_plca_get_status_req_dump {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_plca_get_status_req_dump *
+ethtool_plca_get_status_req_dump_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_plca_get_status_req_dump));
+}
+void
+ethtool_plca_get_status_req_dump_free(struct ethtool_plca_get_status_req_dump *req);
+
+static inline void
+ethtool_plca_get_status_req_dump_set_header_dev_index(struct ethtool_plca_get_status_req_dump *req,
+                                                     __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_plca_get_status_req_dump_set_header_dev_name(struct ethtool_plca_get_status_req_dump *req,
+                                                    const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_plca_get_status_req_dump_set_header_flags(struct ethtool_plca_get_status_req_dump *req,
+                                                 __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_plca_get_status_list {
+       struct ethtool_plca_get_status_list *next;
+       struct ethtool_plca_get_status_rsp obj __attribute__ ((aligned (8)));
+};
+
+void
+ethtool_plca_get_status_list_free(struct ethtool_plca_get_status_list *rsp);
+
+struct ethtool_plca_get_status_list *
+ethtool_plca_get_status_dump(struct ynl_sock *ys,
+                            struct ethtool_plca_get_status_req_dump *req);
+
+/* ============== ETHTOOL_MSG_MM_GET ============== */
+/* ETHTOOL_MSG_MM_GET - do */
+struct ethtool_mm_get_req {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_mm_get_req *ethtool_mm_get_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_mm_get_req));
+}
+void ethtool_mm_get_req_free(struct ethtool_mm_get_req *req);
+
+static inline void
+ethtool_mm_get_req_set_header_dev_index(struct ethtool_mm_get_req *req,
+                                       __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_mm_get_req_set_header_dev_name(struct ethtool_mm_get_req *req,
+                                      const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_mm_get_req_set_header_flags(struct ethtool_mm_get_req *req,
+                                   __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_mm_get_rsp {
+       struct {
+               __u32 header:1;
+               __u32 pmac_enabled:1;
+               __u32 tx_enabled:1;
+               __u32 tx_active:1;
+               __u32 tx_min_frag_size:1;
+               __u32 rx_min_frag_size:1;
+               __u32 verify_enabled:1;
+               __u32 verify_time:1;
+               __u32 max_verify_time:1;
+               __u32 stats:1;
+       } _present;
+
+       struct ethtool_header header;
+       __u8 pmac_enabled;
+       __u8 tx_enabled;
+       __u8 tx_active;
+       __u32 tx_min_frag_size;
+       __u32 rx_min_frag_size;
+       __u8 verify_enabled;
+       __u32 verify_time;
+       __u32 max_verify_time;
+       struct ethtool_mm_stat stats;
+};
+
+void ethtool_mm_get_rsp_free(struct ethtool_mm_get_rsp *rsp);
+
+/*
+ * Get MAC Merge configuration and state
+ */
+struct ethtool_mm_get_rsp *
+ethtool_mm_get(struct ynl_sock *ys, struct ethtool_mm_get_req *req);
+
+/* ETHTOOL_MSG_MM_GET - dump */
+struct ethtool_mm_get_req_dump {
+       struct {
+               __u32 header:1;
+       } _present;
+
+       struct ethtool_header header;
+};
+
+static inline struct ethtool_mm_get_req_dump *
+ethtool_mm_get_req_dump_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_mm_get_req_dump));
+}
+void ethtool_mm_get_req_dump_free(struct ethtool_mm_get_req_dump *req);
+
+static inline void
+ethtool_mm_get_req_dump_set_header_dev_index(struct ethtool_mm_get_req_dump *req,
+                                            __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_mm_get_req_dump_set_header_dev_name(struct ethtool_mm_get_req_dump *req,
+                                           const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_mm_get_req_dump_set_header_flags(struct ethtool_mm_get_req_dump *req,
+                                        __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+
+struct ethtool_mm_get_list {
+       struct ethtool_mm_get_list *next;
+       struct ethtool_mm_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_mm_get_list_free(struct ethtool_mm_get_list *rsp);
+
+struct ethtool_mm_get_list *
+ethtool_mm_get_dump(struct ynl_sock *ys, struct ethtool_mm_get_req_dump *req);
+
+/* ETHTOOL_MSG_MM_GET - notify */
+struct ethtool_mm_get_ntf {
+       __u16 family;
+       __u8 cmd;
+       struct ynl_ntf_base_type *next;
+       void (*free)(struct ethtool_mm_get_ntf *ntf);
+       struct ethtool_mm_get_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_mm_get_ntf_free(struct ethtool_mm_get_ntf *rsp);
+
+/* ============== ETHTOOL_MSG_MM_SET ============== */
+/* ETHTOOL_MSG_MM_SET - do */
+struct ethtool_mm_set_req {
+       struct {
+               __u32 header:1;
+               __u32 verify_enabled:1;
+               __u32 verify_time:1;
+               __u32 tx_enabled:1;
+               __u32 pmac_enabled:1;
+               __u32 tx_min_frag_size:1;
+       } _present;
+
+       struct ethtool_header header;
+       __u8 verify_enabled;
+       __u32 verify_time;
+       __u8 tx_enabled;
+       __u8 pmac_enabled;
+       __u32 tx_min_frag_size;
+};
+
+static inline struct ethtool_mm_set_req *ethtool_mm_set_req_alloc(void)
+{
+       return calloc(1, sizeof(struct ethtool_mm_set_req));
+}
+void ethtool_mm_set_req_free(struct ethtool_mm_set_req *req);
+
+static inline void
+ethtool_mm_set_req_set_header_dev_index(struct ethtool_mm_set_req *req,
+                                       __u32 dev_index)
+{
+       req->_present.header = 1;
+       req->header._present.dev_index = 1;
+       req->header.dev_index = dev_index;
+}
+static inline void
+ethtool_mm_set_req_set_header_dev_name(struct ethtool_mm_set_req *req,
+                                      const char *dev_name)
+{
+       free(req->header.dev_name);
+       req->header._present.dev_name_len = strlen(dev_name);
+       req->header.dev_name = malloc(req->header._present.dev_name_len + 1);
+       memcpy(req->header.dev_name, dev_name, req->header._present.dev_name_len);
+       req->header.dev_name[req->header._present.dev_name_len] = 0;
+}
+static inline void
+ethtool_mm_set_req_set_header_flags(struct ethtool_mm_set_req *req,
+                                   __u32 flags)
+{
+       req->_present.header = 1;
+       req->header._present.flags = 1;
+       req->header.flags = flags;
+}
+static inline void
+ethtool_mm_set_req_set_verify_enabled(struct ethtool_mm_set_req *req,
+                                     __u8 verify_enabled)
+{
+       req->_present.verify_enabled = 1;
+       req->verify_enabled = verify_enabled;
+}
+static inline void
+ethtool_mm_set_req_set_verify_time(struct ethtool_mm_set_req *req,
+                                  __u32 verify_time)
+{
+       req->_present.verify_time = 1;
+       req->verify_time = verify_time;
+}
+static inline void
+ethtool_mm_set_req_set_tx_enabled(struct ethtool_mm_set_req *req,
+                                 __u8 tx_enabled)
+{
+       req->_present.tx_enabled = 1;
+       req->tx_enabled = tx_enabled;
+}
+static inline void
+ethtool_mm_set_req_set_pmac_enabled(struct ethtool_mm_set_req *req,
+                                   __u8 pmac_enabled)
+{
+       req->_present.pmac_enabled = 1;
+       req->pmac_enabled = pmac_enabled;
+}
+static inline void
+ethtool_mm_set_req_set_tx_min_frag_size(struct ethtool_mm_set_req *req,
+                                       __u32 tx_min_frag_size)
+{
+       req->_present.tx_min_frag_size = 1;
+       req->tx_min_frag_size = tx_min_frag_size;
+}
+
+/*
+ * Set MAC Merge configuration
+ */
+int ethtool_mm_set(struct ynl_sock *ys, struct ethtool_mm_set_req *req);
+
+/* ETHTOOL_MSG_CABLE_TEST_NTF - event */
+struct ethtool_cable_test_ntf_rsp {
+       struct {
+               __u32 header:1;
+               __u32 status:1;
+       } _present;
+
+       struct ethtool_header header;
+       __u8 status;
+};
+
+struct ethtool_cable_test_ntf {
+       __u16 family;
+       __u8 cmd;
+       struct ynl_ntf_base_type *next;
+       void (*free)(struct ethtool_cable_test_ntf *ntf);
+       struct ethtool_cable_test_ntf_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_cable_test_ntf_free(struct ethtool_cable_test_ntf *rsp);
+
+/* ETHTOOL_MSG_CABLE_TEST_TDR_NTF - event */
+struct ethtool_cable_test_tdr_ntf_rsp {
+       struct {
+               __u32 header:1;
+               __u32 status:1;
+               __u32 nest:1;
+       } _present;
+
+       struct ethtool_header header;
+       __u8 status;
+       struct ethtool_cable_nest nest;
+};
+
+struct ethtool_cable_test_tdr_ntf {
+       __u16 family;
+       __u8 cmd;
+       struct ynl_ntf_base_type *next;
+       void (*free)(struct ethtool_cable_test_tdr_ntf *ntf);
+       struct ethtool_cable_test_tdr_ntf_rsp obj __attribute__ ((aligned (8)));
+};
+
+void ethtool_cable_test_tdr_ntf_free(struct ethtool_cable_test_tdr_ntf *rsp);
+
+#endif /* _LINUX_ETHTOOL_GEN_H */