diff options
Diffstat (limited to 'src/libsystemd/sd-netlink')
| -rw-r--r-- | src/libsystemd/sd-netlink/netlink-internal.h | 27 | ||||
| -rw-r--r-- | src/libsystemd/sd-netlink/netlink-message.c | 291 | ||||
| -rw-r--r-- | src/libsystemd/sd-netlink/netlink-socket.c | 4 | ||||
| -rw-r--r-- | src/libsystemd/sd-netlink/netlink-types.c | 528 | ||||
| -rw-r--r-- | src/libsystemd/sd-netlink/netlink-types.h | 61 | 
5 files changed, 493 insertions, 418 deletions
| diff --git a/src/libsystemd/sd-netlink/netlink-internal.h b/src/libsystemd/sd-netlink/netlink-internal.h index 7290f4e875..4026e2c341 100644 --- a/src/libsystemd/sd-netlink/netlink-internal.h +++ b/src/libsystemd/sd-netlink/netlink-internal.h @@ -92,18 +92,27 @@ struct sd_netlink {          sd_event *event;  }; +struct netlink_attribute { +        size_t offset; /* offset from hdr to attribute */ +        bool nested:1; +        bool net_byteorder:1; +}; + +struct netlink_container { +        const struct NLTypeSystem *type_system; /* the type system of the container */ +        size_t offset; /* offset from hdr to the start of the container */ +        struct netlink_attribute *attributes; +        unsigned short n_attributes; /* number of attributes in container */ +}; +  struct sd_netlink_message {          RefCount n_ref;          sd_netlink *rtnl;          struct nlmsghdr *hdr; -        const struct NLTypeSystem *(container_type_system[RTNL_CONTAINER_DEPTH]); /* the type of the container and all its parents */ -        size_t container_offsets[RTNL_CONTAINER_DEPTH]; /* offset from hdr to each container's start */ +        struct netlink_container containers[RTNL_CONTAINER_DEPTH];          unsigned n_containers; /* number of containers */ -        size_t next_rta_offset; /* offset from hdr to next rta */ -        size_t *rta_offset_tb[RTNL_CONTAINER_DEPTH]; -        unsigned short rta_tb_size[RTNL_CONTAINER_DEPTH];          bool sealed:1;          bool broadcast:1; @@ -122,14 +131,6 @@ int socket_read_message(sd_netlink *nl);  int rtnl_rqueue_make_room(sd_netlink *rtnl);  int rtnl_rqueue_partial_make_room(sd_netlink *rtnl); -int rtnl_message_read_internal(sd_netlink_message *m, unsigned short type, void **data); -int rtnl_message_parse(sd_netlink_message *m, -                       size_t **rta_offset_tb, -                       unsigned short *rta_tb_size, -                       int max, -                       struct rtattr *rta, -                       unsigned int rt_len); -  /* Make sure callbacks don't destroy the rtnl connection */  #define RTNL_DONT_DESTROY(rtnl) \          _cleanup_netlink_unref_ _unused_ sd_netlink *_dont_destroy_##rtnl = sd_netlink_ref(rtnl) diff --git a/src/libsystemd/sd-netlink/netlink-message.c b/src/libsystemd/sd-netlink/netlink-message.c index 87324fc2f7..b0ed2f2882 100644 --- a/src/libsystemd/sd-netlink/netlink-message.c +++ b/src/libsystemd/sd-netlink/netlink-message.c @@ -34,10 +34,11 @@  #include "netlink-internal.h"  #include "netlink-types.h" -#define GET_CONTAINER(m, i) ((i) < (m)->n_containers ? (struct rtattr*)((uint8_t*)(m)->hdr + (m)->container_offsets[i]) : NULL) +#define GET_CONTAINER(m, i) ((i) < (m)->n_containers ? (struct rtattr*)((uint8_t*)(m)->hdr + (m)->containers[i].offset) : NULL)  #define PUSH_CONTAINER(m, new) (m)->container_offsets[(m)->n_containers ++] = (uint8_t*)(new) - (uint8_t*)(m)->hdr;  #define RTA_TYPE(rta) ((rta)->rta_type & NLA_TYPE_MASK) +#define RTA_FLAGS(rta) ((rta)->rta_type & ~NLA_TYPE_MASK)  int message_new_empty(sd_netlink *rtnl, sd_netlink_message **ret) {          sd_netlink_message *m; @@ -68,15 +69,18 @@ int message_new(sd_netlink *rtnl, sd_netlink_message **ret, uint16_t type) {          size_t size;          int r; -        r = type_system_get_type(NULL, &nl_type, type); +        r = type_system_get_type(&type_system_root, &nl_type, type);          if (r < 0)                  return r; +        if (type_get_type(nl_type) != NETLINK_TYPE_NESTED) +                return -EINVAL; +          r = message_new_empty(rtnl, &m);          if (r < 0)                  return r; -        size = NLMSG_SPACE(nl_type->size); +        size = NLMSG_SPACE(type_get_size(nl_type));          assert(size >= sizeof(struct nlmsghdr));          m->hdr = malloc0(size); @@ -85,7 +89,7 @@ int message_new(sd_netlink *rtnl, sd_netlink_message **ret, uint16_t type) {          m->hdr->nlmsg_flags = NLM_F_REQUEST | NLM_F_ACK; -        m->container_type_system[0] = nl_type->type_system; +        type_get_type_system(nl_type, &m->containers[0].type_system);          m->hdr->nlmsg_len = size;          m->hdr->nlmsg_type = type; @@ -126,7 +130,7 @@ sd_netlink_message *sd_netlink_message_unref(sd_netlink_message *m) {                  free(m->hdr);                  for (i = 0; i <= m->n_containers; i++) -                        free(m->rta_offset_tb[i]); +                        free(m->containers[i].attributes);                  sd_netlink_message_unref(m->next); @@ -214,18 +218,22 @@ static int add_rtattr(sd_netlink_message *m, unsigned short type, const void *da          return offset;  } -static int message_attribute_has_type(sd_netlink_message *m, uint16_t attribute_type, uint16_t data_type) { +static int message_attribute_has_type(sd_netlink_message *m, size_t *out_size, uint16_t attribute_type, uint16_t data_type) {          const NLType *type;          int r; -        r = type_system_get_type(m->container_type_system[m->n_containers], &type, attribute_type); +        assert(m); + +        r = type_system_get_type(m->containers[m->n_containers].type_system, &type, attribute_type);          if (r < 0)                  return r; -        if (type->type != data_type) +        if (type_get_type(type) != data_type)                  return -EINVAL; -        return type->size; +        if (out_size) +                *out_size = type_get_size(type); +        return 0;  }  int sd_netlink_message_append_string(sd_netlink_message *m, unsigned short type, const char *data) { @@ -236,11 +244,9 @@ int sd_netlink_message_append_string(sd_netlink_message *m, unsigned short type,          assert_return(!m->sealed, -EPERM);          assert_return(data, -EINVAL); -        r = message_attribute_has_type(m, type, NLA_STRING); +        r = message_attribute_has_type(m, &size, type, NETLINK_TYPE_STRING);          if (r < 0)                  return r; -        else -                size = (size_t)r;          if (size) {                  length = strnlen(data, size+1); @@ -262,7 +268,7 @@ int sd_netlink_message_append_u8(sd_netlink_message *m, unsigned short type, uin          assert_return(m, -EINVAL);          assert_return(!m->sealed, -EPERM); -        r = message_attribute_has_type(m, type, NLA_U8); +        r = message_attribute_has_type(m, NULL, type, NETLINK_TYPE_U8);          if (r < 0)                  return r; @@ -280,7 +286,7 @@ int sd_netlink_message_append_u16(sd_netlink_message *m, unsigned short type, ui          assert_return(m, -EINVAL);          assert_return(!m->sealed, -EPERM); -        r = message_attribute_has_type(m, type, NLA_U16); +        r = message_attribute_has_type(m, NULL, type, NETLINK_TYPE_U16);          if (r < 0)                  return r; @@ -297,7 +303,7 @@ int sd_netlink_message_append_u32(sd_netlink_message *m, unsigned short type, ui          assert_return(m, -EINVAL);          assert_return(!m->sealed, -EPERM); -        r = message_attribute_has_type(m, type, NLA_U32); +        r = message_attribute_has_type(m, NULL, type, NETLINK_TYPE_U32);          if (r < 0)                  return r; @@ -315,7 +321,7 @@ int sd_netlink_message_append_in_addr(sd_netlink_message *m, unsigned short type          assert_return(!m->sealed, -EPERM);          assert_return(data, -EINVAL); -        r = message_attribute_has_type(m, type, NLA_IN_ADDR); +        r = message_attribute_has_type(m, NULL, type, NETLINK_TYPE_IN_ADDR);          if (r < 0)                  return r; @@ -333,7 +339,7 @@ int sd_netlink_message_append_in6_addr(sd_netlink_message *m, unsigned short typ          assert_return(!m->sealed, -EPERM);          assert_return(data, -EINVAL); -        r = message_attribute_has_type(m, type, NLA_IN_ADDR); +        r = message_attribute_has_type(m, NULL, type, NETLINK_TYPE_IN_ADDR);          if (r < 0)                  return r; @@ -351,7 +357,7 @@ int sd_netlink_message_append_ether_addr(sd_netlink_message *m, unsigned short t          assert_return(!m->sealed, -EPERM);          assert_return(data, -EINVAL); -        r = message_attribute_has_type(m, type, NLA_ETHER_ADDR); +        r = message_attribute_has_type(m, NULL, type, NETLINK_TYPE_ETHER_ADDR);          if (r < 0)                  return r; @@ -369,7 +375,7 @@ int sd_netlink_message_append_cache_info(sd_netlink_message *m, unsigned short t          assert_return(!m->sealed, -EPERM);          assert_return(info, -EINVAL); -        r = message_attribute_has_type(m, type, NLA_CACHE_INFO); +        r = message_attribute_has_type(m, NULL, type, NETLINK_TYPE_CACHE_INFO);          if (r < 0)                  return r; @@ -388,34 +394,31 @@ int sd_netlink_message_open_container(sd_netlink_message *m, unsigned short type          assert_return(!m->sealed, -EPERM);          assert_return(m->n_containers < RTNL_CONTAINER_DEPTH, -ERANGE); -        r = message_attribute_has_type(m, type, NLA_NESTED); +        r = message_attribute_has_type(m, &size, type, NETLINK_TYPE_NESTED);          if (r < 0) {                  const NLTypeSystemUnion *type_system_union;                  int family; -                r = message_attribute_has_type(m, type, NLA_UNION); +                r = message_attribute_has_type(m, &size, type, NETLINK_TYPE_UNION);                  if (r < 0)                          return r; -                size = (size_t) r;                  r = sd_rtnl_message_get_family(m, &family);                  if (r < 0)                          return r; -                r = type_system_get_type_system_union(m->container_type_system[m->n_containers], &type_system_union, type); +                r = type_system_get_type_system_union(m->containers[m->n_containers].type_system, &type_system_union, type);                  if (r < 0)                          return r;                  r = type_system_union_protocol_get_type_system(type_system_union, -                                                               &m->container_type_system[m->n_containers + 1], +                                                               &m->containers[m->n_containers + 1].type_system,                                                                 family);                  if (r < 0)                          return r;          } else { -                size = (size_t)r; - -                r = type_system_get_type_system(m->container_type_system[m->n_containers], -                                                &m->container_type_system[m->n_containers + 1], +                r = type_system_get_type_system(m->containers[m->n_containers].type_system, +                                                &m->containers[m->n_containers + 1].type_system,                                                  type);                  if (r < 0)                          return r; @@ -425,7 +428,7 @@ int sd_netlink_message_open_container(sd_netlink_message *m, unsigned short type          if (r < 0)                  return r; -        m->container_offsets[m->n_containers ++] = r; +        m->containers[m->n_containers ++].offset = r;          return 0;  } @@ -437,12 +440,12 @@ int sd_netlink_message_open_container_union(sd_netlink_message *m, unsigned shor          assert_return(m, -EINVAL);          assert_return(!m->sealed, -EPERM); -        r = type_system_get_type_system_union(m->container_type_system[m->n_containers], &type_system_union, type); +        r = type_system_get_type_system_union(m->containers[m->n_containers].type_system, &type_system_union, type);          if (r < 0)                  return r;          r = type_system_union_get_type_system(type_system_union, -                                              &m->container_type_system[m->n_containers + 1], +                                              &m->containers[m->n_containers + 1].type_system,                                                key);          if (r < 0)                  return r; @@ -452,11 +455,11 @@ int sd_netlink_message_open_container_union(sd_netlink_message *m, unsigned shor                  return r;          /* do we evere need non-null size */ -        r = add_rtattr(m, type, NULL, 0); +        r = add_rtattr(m, type | NLA_F_NESTED, NULL, 0);          if (r < 0)                  return r; -        m->container_offsets[m->n_containers ++] = r; +        m->containers[m->n_containers ++].offset = r;          return 0;  } @@ -467,29 +470,35 @@ int sd_netlink_message_close_container(sd_netlink_message *m) {          assert_return(!m->sealed, -EPERM);          assert_return(m->n_containers > 0, -EINVAL); -        m->container_type_system[m->n_containers] = NULL; +        m->containers[m->n_containers].type_system = NULL;          m->n_containers --;          return 0;  } -int rtnl_message_read_internal(sd_netlink_message *m, unsigned short type, void **data) { +static int netlink_message_read_internal(sd_netlink_message *m, unsigned short type, void **data, bool *net_byteorder) { +        struct netlink_attribute *attribute;          struct rtattr *rta;          assert_return(m, -EINVAL);          assert_return(m->sealed, -EPERM);          assert_return(data, -EINVAL);          assert(m->n_containers <= RTNL_CONTAINER_DEPTH); -        assert(m->rta_offset_tb[m->n_containers]); -        assert(type < m->rta_tb_size[m->n_containers]); +        assert(m->containers[m->n_containers].attributes); +        assert(type < m->containers[m->n_containers].n_attributes); + +        attribute = &m->containers[m->n_containers].attributes[type]; -        if(!m->rta_offset_tb[m->n_containers][type]) +        if(!attribute->offset)                  return -ENODATA; -        rta = (struct rtattr*)((uint8_t *) m->hdr + m->rta_offset_tb[m->n_containers][type]); +        rta = (struct rtattr*)((uint8_t *) m->hdr + attribute->offset);          *data = RTA_DATA(rta); +        if (net_byteorder) +                *net_byteorder = attribute->net_byteorder; +          return RTA_PAYLOAD(rta);  } @@ -499,11 +508,11 @@ int sd_netlink_message_read_string(sd_netlink_message *m, unsigned short type, c          assert_return(m, -EINVAL); -        r = message_attribute_has_type(m, type, NLA_STRING); +        r = message_attribute_has_type(m, NULL, type, NETLINK_TYPE_STRING);          if (r < 0)                  return r; -        r = rtnl_message_read_internal(m, type, &attr_data); +        r = netlink_message_read_internal(m, type, &attr_data, NULL);          if (r < 0)                  return r;          else if (strnlen(attr_data, r) >= (size_t) r) @@ -521,11 +530,11 @@ int sd_netlink_message_read_u8(sd_netlink_message *m, unsigned short type, uint8          assert_return(m, -EINVAL); -        r = message_attribute_has_type(m, type, NLA_U8); +        r = message_attribute_has_type(m, NULL, type, NETLINK_TYPE_U8);          if (r < 0)                  return r; -        r = rtnl_message_read_internal(m, type, &attr_data); +        r = netlink_message_read_internal(m, type, &attr_data, NULL);          if (r < 0)                  return r;          else if ((size_t) r < sizeof(uint8_t)) @@ -538,45 +547,55 @@ int sd_netlink_message_read_u8(sd_netlink_message *m, unsigned short type, uint8  }  int sd_netlink_message_read_u16(sd_netlink_message *m, unsigned short type, uint16_t *data) { -        int r;          void *attr_data; +        bool net_byteorder; +        int r;          assert_return(m, -EINVAL); -        r = message_attribute_has_type(m, type, NLA_U16); +        r = message_attribute_has_type(m, NULL, type, NETLINK_TYPE_U16);          if (r < 0)                  return r; -        r = rtnl_message_read_internal(m, type, &attr_data); +        r = netlink_message_read_internal(m, type, &attr_data, &net_byteorder);          if (r < 0)                  return r;          else if ((size_t) r < sizeof(uint16_t))                  return -EIO; -        if (data) -                *data = *(uint16_t *) attr_data; +        if (data) { +                if (net_byteorder) +                        *data = be16toh(*(uint16_t *) attr_data); +                else +                        *data = *(uint16_t *) attr_data; +        }          return 0;  }  int sd_netlink_message_read_u32(sd_netlink_message *m, unsigned short type, uint32_t *data) { -        int r;          void *attr_data; +        bool net_byteorder; +        int r;          assert_return(m, -EINVAL); -        r = message_attribute_has_type(m, type, NLA_U32); +        r = message_attribute_has_type(m, NULL, type, NETLINK_TYPE_U32);          if (r < 0)                  return r; -        r = rtnl_message_read_internal(m, type, &attr_data); +        r = netlink_message_read_internal(m, type, &attr_data, &net_byteorder);          if (r < 0)                  return r;          else if ((size_t)r < sizeof(uint32_t))                  return -EIO; -        if (data) -                *data = *(uint32_t *) attr_data; +        if (data) { +                if (net_byteorder) +                        *data = be32toh(*(uint32_t *) attr_data); +                else +                        *data = *(uint32_t *) attr_data; +        }          return 0;  } @@ -587,11 +606,11 @@ int sd_netlink_message_read_ether_addr(sd_netlink_message *m, unsigned short typ          assert_return(m, -EINVAL); -        r = message_attribute_has_type(m, type, NLA_ETHER_ADDR); +        r = message_attribute_has_type(m, NULL, type, NETLINK_TYPE_ETHER_ADDR);          if (r < 0)                  return r; -        r = rtnl_message_read_internal(m, type, &attr_data); +        r = netlink_message_read_internal(m, type, &attr_data, NULL);          if (r < 0)                  return r;          else if ((size_t)r < sizeof(struct ether_addr)) @@ -609,11 +628,11 @@ int sd_netlink_message_read_cache_info(sd_netlink_message *m, unsigned short typ          assert_return(m, -EINVAL); -        r = message_attribute_has_type(m, type, NLA_CACHE_INFO); +        r = message_attribute_has_type(m, NULL, type, NETLINK_TYPE_CACHE_INFO);          if (r < 0)                  return r; -        r = rtnl_message_read_internal(m, type, &attr_data); +        r = netlink_message_read_internal(m, type, &attr_data, NULL);          if (r < 0)                  return r;          else if ((size_t)r < sizeof(struct ifa_cacheinfo)) @@ -631,11 +650,11 @@ int sd_netlink_message_read_in_addr(sd_netlink_message *m, unsigned short type,          assert_return(m, -EINVAL); -        r = message_attribute_has_type(m, type, NLA_IN_ADDR); +        r = message_attribute_has_type(m, NULL, type, NETLINK_TYPE_IN_ADDR);          if (r < 0)                  return r; -        r = rtnl_message_read_internal(m, type, &attr_data); +        r = netlink_message_read_internal(m, type, &attr_data, NULL);          if (r < 0)                  return r;          else if ((size_t)r < sizeof(struct in_addr)) @@ -653,11 +672,11 @@ int sd_netlink_message_read_in6_addr(sd_netlink_message *m, unsigned short type,          assert_return(m, -EINVAL); -        r = message_attribute_has_type(m, type, NLA_IN_ADDR); +        r = message_attribute_has_type(m, NULL, type, NETLINK_TYPE_IN_ADDR);          if (r < 0)                  return r; -        r = rtnl_message_read_internal(m, type, &attr_data); +        r = netlink_message_read_internal(m, type, &attr_data, NULL);          if (r < 0)                  return r;          else if ((size_t)r < sizeof(struct in6_addr)) @@ -669,34 +688,73 @@ int sd_netlink_message_read_in6_addr(sd_netlink_message *m, unsigned short type,          return 0;  } -int sd_netlink_message_enter_container(sd_netlink_message *m, unsigned short type) { +static int netlink_container_parse(sd_netlink_message *m, +                                   struct netlink_container *container, +                                   int count, +                                   struct rtattr *rta, +                                   unsigned int rt_len) { +        _cleanup_free_ struct netlink_attribute *attributes = NULL; + +        attributes = new0(struct netlink_attribute, count); +        if(!attributes) +                return -ENOMEM; + +        for (; RTA_OK(rta, rt_len); rta = RTA_NEXT(rta, rt_len)) { +                unsigned short type; + +                type = RTA_TYPE(rta); + +                /* if the kernel is newer than the headers we used +                   when building, we ignore out-of-range attributes */ +                if (type >= count) +                        continue; + +                if (attributes[type].offset) +                        log_debug("rtnl: message parse - overwriting repeated attribute"); + +                attributes[type].offset = (uint8_t *) rta - (uint8_t *) m->hdr; +                attributes[type].nested = RTA_FLAGS(rta) & NLA_F_NESTED; +                attributes[type].net_byteorder = RTA_FLAGS(rta) & NLA_F_NET_BYTEORDER; +        } + +        container->attributes = attributes; +        attributes = NULL; +        container->n_attributes = count; + +        return 0; +} + +int sd_netlink_message_enter_container(sd_netlink_message *m, unsigned short type_id) {          const NLType *nl_type;          const NLTypeSystem *type_system;          void *container; +        uint16_t type;          size_t size;          int r;          assert_return(m, -EINVAL);          assert_return(m->n_containers < RTNL_CONTAINER_DEPTH, -EINVAL); -        r = type_system_get_type(m->container_type_system[m->n_containers], +        r = type_system_get_type(m->containers[m->n_containers].type_system,                                   &nl_type, -                                 type); +                                 type_id);          if (r < 0)                  return r; -        if (nl_type->type == NLA_NESTED) { -                r = type_system_get_type_system(m->container_type_system[m->n_containers], +        type = type_get_type(nl_type); + +        if (type == NETLINK_TYPE_NESTED) { +                r = type_system_get_type_system(m->containers[m->n_containers].type_system,                                                  &type_system, -                                                type); +                                                type_id);                  if (r < 0)                          return r; -        } else if (nl_type->type == NLA_UNION) { +        } else if (type == NETLINK_TYPE_UNION) {                  const NLTypeSystemUnion *type_system_union; -                r = type_system_get_type_system_union(m->container_type_system[m->n_containers], +                r = type_system_get_type_system_union(m->containers[m->n_containers].type_system,                                                        &type_system_union, -                                                      type); +                                                      type_id);                  if (r < 0)                          return r; @@ -739,7 +797,7 @@ int sd_netlink_message_enter_container(sd_netlink_message *m, unsigned short typ          } else                  return -EINVAL; -        r = rtnl_message_read_internal(m, type, &container); +        r = netlink_message_read_internal(m, type_id, &container, NULL);          if (r < 0)                  return r;          else @@ -747,18 +805,17 @@ int sd_netlink_message_enter_container(sd_netlink_message *m, unsigned short typ          m->n_containers ++; -        r = rtnl_message_parse(m, -                               &m->rta_offset_tb[m->n_containers], -                               &m->rta_tb_size[m->n_containers], -                               type_system->max, -                               container, -                               size); +        r = netlink_container_parse(m, +                                    &m->containers[m->n_containers], +                                    type_system_get_count(type_system), +                                    container, +                                    size);          if (r < 0) {                  m->n_containers --;                  return r;          } -        m->container_type_system[m->n_containers] = type_system; +        m->containers[m->n_containers].type_system = type_system;          return 0;  } @@ -768,9 +825,9 @@ int sd_netlink_message_exit_container(sd_netlink_message *m) {          assert_return(m->sealed, -EINVAL);          assert_return(m->n_containers > 0, -EINVAL); -        free(m->rta_offset_tb[m->n_containers]); -        m->rta_offset_tb[m->n_containers] = NULL; -        m->container_type_system[m->n_containers] = NULL; +        free(m->containers[m->n_containers].attributes); +        m->containers[m->n_containers].attributes = NULL; +        m->containers[m->n_containers].type_system = NULL;          m->n_containers --; @@ -805,43 +862,10 @@ int sd_netlink_message_get_errno(sd_netlink_message *m) {          return err->error;  } -int rtnl_message_parse(sd_netlink_message *m, -                       size_t **rta_offset_tb, -                       unsigned short *rta_tb_size, -                       int max, -                       struct rtattr *rta, -                       unsigned int rt_len) { -        unsigned short type; -        size_t *tb; - -        tb = new0(size_t, max + 1); -        if(!tb) -                return -ENOMEM; - -        *rta_tb_size = max + 1; - -        for (; RTA_OK(rta, rt_len); rta = RTA_NEXT(rta, rt_len)) { -                type = RTA_TYPE(rta); - -                /* if the kernel is newer than the headers we used -                   when building, we ignore out-of-range attributes -                 */ -                if (type > max) -                        continue; - -                if (tb[type]) -                        log_debug("rtnl: message parse - overwriting repeated attribute"); - -                tb[type] = (uint8_t *) rta - (uint8_t *) m->hdr; -        } - -        *rta_offset_tb = tb; - -        return 0; -} -  int sd_netlink_message_rewind(sd_netlink_message *m) { -        const NLType *type; +        const NLType *nl_type; +        uint16_t type; +        size_t size;          unsigned i;          int r; @@ -852,39 +876,38 @@ int sd_netlink_message_rewind(sd_netlink_message *m) {                  rtnl_message_seal(m);          for (i = 1; i <= m->n_containers; i++) { -                free(m->rta_offset_tb[i]); -                m->rta_offset_tb[i] = NULL; -                m->rta_tb_size[i] = 0; -                m->container_type_system[i] = NULL; +                free(m->containers[i].attributes); +                m->containers[i].attributes = NULL;          }          m->n_containers = 0; -        if (m->rta_offset_tb[0]) { +        if (m->containers[0].attributes) {                  /* top-level attributes have already been parsed */                  return 0;          }          assert(m->hdr); -        r = type_system_get_type(NULL, &type, m->hdr->nlmsg_type); +        r = type_system_get_type(&type_system_root, &nl_type, m->hdr->nlmsg_type);          if (r < 0)                  return r; -        if (type->type == NLA_NESTED) { -                const NLTypeSystem *type_system = type->type_system; +        type = type_get_type(nl_type); +        size = type_get_size(nl_type); + +        if (type == NETLINK_TYPE_NESTED) { +                const NLTypeSystem *type_system; -                assert(type_system); +                type_get_type_system(nl_type, &type_system); -                m->container_type_system[0] = type_system; +                m->containers[0].type_system = type_system; -                r = rtnl_message_parse(m, -                                       &m->rta_offset_tb[m->n_containers], -                                       &m->rta_tb_size[m->n_containers], -                                       type_system->max, -                                       (struct rtattr*)((uint8_t*)NLMSG_DATA(m->hdr) + -                                                        NLMSG_ALIGN(type->size)), -                                       NLMSG_PAYLOAD(m->hdr, type->size)); +                r = netlink_container_parse(m, +                                            &m->containers[m->n_containers], +                                            type_system_get_count(type_system), +                                            (struct rtattr*)((uint8_t*)NLMSG_DATA(m->hdr) + NLMSG_ALIGN(size)), +                                            NLMSG_PAYLOAD(m->hdr, size));                  if (r < 0)                          return r;          } diff --git a/src/libsystemd/sd-netlink/netlink-socket.c b/src/libsystemd/sd-netlink/netlink-socket.c index 8136cf36ae..84ff7c38c9 100644 --- a/src/libsystemd/sd-netlink/netlink-socket.c +++ b/src/libsystemd/sd-netlink/netlink-socket.c @@ -243,7 +243,7 @@ int socket_read_message(sd_netlink *rtnl) {                  }                  /* check that we support this message type */ -                r = type_system_get_type(NULL, &nl_type, new_msg->nlmsg_type); +                r = type_system_get_type(&type_system_root, &nl_type, new_msg->nlmsg_type);                  if (r < 0) {                          if (r == -EOPNOTSUPP)                                  log_debug("sd-netlink: ignored message with unknown type: %i", @@ -253,7 +253,7 @@ int socket_read_message(sd_netlink *rtnl) {                  }                  /* check that the size matches the message type */ -                if (new_msg->nlmsg_len < NLMSG_LENGTH(nl_type->size)) { +                if (new_msg->nlmsg_len < NLMSG_LENGTH(type_get_size(nl_type))) {                          log_debug("sd-netlink: message larger than expected, dropping");                          continue;                  } diff --git a/src/libsystemd/sd-netlink/netlink-types.c b/src/libsystemd/sd-netlink/netlink-types.c index 273033770f..9b68935245 100644 --- a/src/libsystemd/sd-netlink/netlink-types.c +++ b/src/libsystemd/sd-netlink/netlink-types.c @@ -39,152 +39,205 @@  #include "netlink-types.h"  #include "missing.h" +/* Maximum ARP IP target defined in kernel */ +#define BOND_MAX_ARP_TARGETS    16 + +typedef enum { +        BOND_ARP_TARGETS_0, +        BOND_ARP_TARGETS_1, +        BOND_ARP_TARGETS_2, +        BOND_ARP_TARGETS_3, +        BOND_ARP_TARGETS_4, +        BOND_ARP_TARGETS_5, +        BOND_ARP_TARGETS_6, +        BOND_ARP_TARGETS_7, +        BOND_ARP_TARGETS_8, +        BOND_ARP_TARGETS_9, +        BOND_ARP_TARGETS_10, +        BOND_ARP_TARGETS_11, +        BOND_ARP_TARGETS_12, +        BOND_ARP_TARGETS_13, +        BOND_ARP_TARGETS_14, +        BOND_ARP_TARGETS_MAX = BOND_MAX_ARP_TARGETS, +} BondArpTargets; + +struct NLType { +        uint16_t type; +        size_t size; +        const NLTypeSystem *type_system; +        const NLTypeSystemUnion *type_system_union; +}; + +struct NLTypeSystem { +        uint16_t count; +        const NLType *types; +}; +  static const NLTypeSystem rtnl_link_type_system; +static const NLType empty_types[1] = { +        /* fake array to avoid .types==NULL, which denotes invalid type-systems */ +}; + +static const NLTypeSystem empty_type_system = { +        .count = 0, +        .types = empty_types, +}; +  static const NLType rtnl_link_info_data_veth_types[VETH_INFO_MAX + 1] = { -        [VETH_INFO_PEER]  = { .type = NLA_NESTED, .type_system = &rtnl_link_type_system, .size = sizeof(struct ifinfomsg) }, +        [VETH_INFO_PEER]  = { .type = NETLINK_TYPE_NESTED, .type_system = &rtnl_link_type_system, .size = sizeof(struct ifinfomsg) },  };  static const NLType rtnl_link_info_data_ipvlan_types[IFLA_IPVLAN_MAX + 1] = { -        [IFLA_IPVLAN_MODE]  = { .type = NLA_U16 }, +        [IFLA_IPVLAN_MODE]  = { .type = NETLINK_TYPE_U16 },  };  static const NLType rtnl_link_info_data_macvlan_types[IFLA_MACVLAN_MAX + 1] = { -        [IFLA_MACVLAN_MODE]  = { .type = NLA_U32 }, -        [IFLA_MACVLAN_FLAGS] = { .type = NLA_U16 }, +        [IFLA_MACVLAN_MODE]  = { .type = NETLINK_TYPE_U32 }, +        [IFLA_MACVLAN_FLAGS] = { .type = NETLINK_TYPE_U16 },  };  static const NLType rtnl_link_info_data_bridge_types[IFLA_BRIDGE_MAX + 1] = { -        [IFLA_BRIDGE_FLAGS]     = { .type = NLA_U16 }, -        [IFLA_BRIDGE_MODE]      = { .type = NLA_U16 }, +        [IFLA_BRIDGE_FLAGS]     = { .type = NETLINK_TYPE_U16 }, +        [IFLA_BRIDGE_MODE]      = { .type = NETLINK_TYPE_U16 },  /* -        [IFLA_BRIDGE_VLAN_INFO] = { .type = NLA_BINARY, +        [IFLA_BRIDGE_VLAN_INFO] = { .type = NETLINK_TYPE_BINARY,                                      .len = sizeof(struct bridge_vlan_info), },  */  };  static const NLType rtnl_link_info_data_vlan_types[IFLA_VLAN_MAX + 1] = { -        [IFLA_VLAN_ID]          = { .type = NLA_U16 }, +        [IFLA_VLAN_ID]          = { .type = NETLINK_TYPE_U16 },  /*          [IFLA_VLAN_FLAGS]       = { .len = sizeof(struct ifla_vlan_flags) }, -        [IFLA_VLAN_EGRESS_QOS]  = { .type = NLA_NESTED }, -        [IFLA_VLAN_INGRESS_QOS] = { .type = NLA_NESTED }, +        [IFLA_VLAN_EGRESS_QOS]  = { .type = NETLINK_TYPE_NESTED }, +        [IFLA_VLAN_INGRESS_QOS] = { .type = NETLINK_TYPE_NESTED },  */ -        [IFLA_VLAN_PROTOCOL]    = { .type = NLA_U16 }, +        [IFLA_VLAN_PROTOCOL]    = { .type = NETLINK_TYPE_U16 },  };  static const NLType rtnl_link_info_data_vxlan_types[IFLA_VXLAN_MAX+1] = { -        [IFLA_VXLAN_ID] = { .type = NLA_U32 }, -        [IFLA_VXLAN_GROUP] = {.type = NLA_IN_ADDR }, -        [IFLA_VXLAN_LINK] = { .type = NLA_U32 }, -        [IFLA_VXLAN_LOCAL] = { .type = NLA_U32}, -        [IFLA_VXLAN_TTL] = { .type = NLA_U8 }, -        [IFLA_VXLAN_TOS] = { .type = NLA_U8 }, -        [IFLA_VXLAN_LEARNING] = { .type = NLA_U8 }, -        [IFLA_VXLAN_AGEING] = { .type = NLA_U32 }, -        [IFLA_VXLAN_LIMIT] = { .type = NLA_U32 }, -        [IFLA_VXLAN_PORT_RANGE] = { .type = NLA_U32}, -        [IFLA_VXLAN_PROXY] = { .type = NLA_U8 }, -        [IFLA_VXLAN_RSC] = { .type = NLA_U8 }, -        [IFLA_VXLAN_L2MISS] = { .type = NLA_U8 }, -        [IFLA_VXLAN_L3MISS] = { .type = NLA_U8 }, +        [IFLA_VXLAN_ID]         = { .type = NETLINK_TYPE_U32 }, +        [IFLA_VXLAN_GROUP]      = { .type = NETLINK_TYPE_IN_ADDR }, +        [IFLA_VXLAN_LINK]       = { .type = NETLINK_TYPE_U32 }, +        [IFLA_VXLAN_LOCAL]      = { .type = NETLINK_TYPE_U32}, +        [IFLA_VXLAN_TTL]        = { .type = NETLINK_TYPE_U8 }, +        [IFLA_VXLAN_TOS]        = { .type = NETLINK_TYPE_U8 }, +        [IFLA_VXLAN_LEARNING]   = { .type = NETLINK_TYPE_U8 }, +        [IFLA_VXLAN_AGEING]     = { .type = NETLINK_TYPE_U32 }, +        [IFLA_VXLAN_LIMIT]      = { .type = NETLINK_TYPE_U32 }, +        [IFLA_VXLAN_PORT_RANGE] = { .type = NETLINK_TYPE_U32}, +        [IFLA_VXLAN_PROXY]      = { .type = NETLINK_TYPE_U8 }, +        [IFLA_VXLAN_RSC]        = { .type = NETLINK_TYPE_U8 }, +        [IFLA_VXLAN_L2MISS]     = { .type = NETLINK_TYPE_U8 }, +        [IFLA_VXLAN_L3MISS]     = { .type = NETLINK_TYPE_U8 },  };  static const NLType rtnl_bond_arp_target_types[BOND_ARP_TARGETS_MAX + 1] = { -        [BOND_ARP_TARGETS_0]        = { .type = NLA_U32 }, -        [BOND_ARP_TARGETS_1]        = { .type = NLA_U32 }, -        [BOND_ARP_TARGETS_2]        = { .type = NLA_U32 }, -        [BOND_ARP_TARGETS_3]        = { .type = NLA_U32 }, -        [BOND_ARP_TARGETS_4]        = { .type = NLA_U32 }, -        [BOND_ARP_TARGETS_5]        = { .type = NLA_U32 }, -        [BOND_ARP_TARGETS_6]        = { .type = NLA_U32 }, -        [BOND_ARP_TARGETS_7]        = { .type = NLA_U32 }, -        [BOND_ARP_TARGETS_8]        = { .type = NLA_U32 }, -        [BOND_ARP_TARGETS_9]        = { .type = NLA_U32 }, -        [BOND_ARP_TARGETS_10]       = { .type = NLA_U32 }, -        [BOND_ARP_TARGETS_11]       = { .type = NLA_U32 }, -        [BOND_ARP_TARGETS_12]       = { .type = NLA_U32 }, -        [BOND_ARP_TARGETS_13]       = { .type = NLA_U32 }, -        [BOND_ARP_TARGETS_14]       = { .type = NLA_U32 }, -        [BOND_ARP_TARGETS_MAX]      = { .type = NLA_U32 }, +        [BOND_ARP_TARGETS_0]        = { .type = NETLINK_TYPE_U32 }, +        [BOND_ARP_TARGETS_1]        = { .type = NETLINK_TYPE_U32 }, +        [BOND_ARP_TARGETS_2]        = { .type = NETLINK_TYPE_U32 }, +        [BOND_ARP_TARGETS_3]        = { .type = NETLINK_TYPE_U32 }, +        [BOND_ARP_TARGETS_4]        = { .type = NETLINK_TYPE_U32 }, +        [BOND_ARP_TARGETS_5]        = { .type = NETLINK_TYPE_U32 }, +        [BOND_ARP_TARGETS_6]        = { .type = NETLINK_TYPE_U32 }, +        [BOND_ARP_TARGETS_7]        = { .type = NETLINK_TYPE_U32 }, +        [BOND_ARP_TARGETS_8]        = { .type = NETLINK_TYPE_U32 }, +        [BOND_ARP_TARGETS_9]        = { .type = NETLINK_TYPE_U32 }, +        [BOND_ARP_TARGETS_10]       = { .type = NETLINK_TYPE_U32 }, +        [BOND_ARP_TARGETS_11]       = { .type = NETLINK_TYPE_U32 }, +        [BOND_ARP_TARGETS_12]       = { .type = NETLINK_TYPE_U32 }, +        [BOND_ARP_TARGETS_13]       = { .type = NETLINK_TYPE_U32 }, +        [BOND_ARP_TARGETS_14]       = { .type = NETLINK_TYPE_U32 }, +        [BOND_ARP_TARGETS_MAX]      = { .type = NETLINK_TYPE_U32 },  };  static const NLTypeSystem rtnl_bond_arp_type_system = { -        .max = ELEMENTSOF(rtnl_bond_arp_target_types) - 1, +        .count = ELEMENTSOF(rtnl_bond_arp_target_types),          .types = rtnl_bond_arp_target_types,  };  static const NLType rtnl_link_info_data_bond_types[IFLA_BOND_MAX + 1] = { -        [IFLA_BOND_MODE]                = { .type = NLA_U8 }, -        [IFLA_BOND_ACTIVE_SLAVE]        = { .type = NLA_U32 }, -        [IFLA_BOND_MIIMON]              = { .type = NLA_U32 }, -        [IFLA_BOND_UPDELAY]             = { .type = NLA_U32 }, -        [IFLA_BOND_DOWNDELAY]           = { .type = NLA_U32 }, -        [IFLA_BOND_USE_CARRIER]         = { .type = NLA_U8 }, -        [IFLA_BOND_ARP_INTERVAL]        = { .type = NLA_U32 }, -        [IFLA_BOND_ARP_IP_TARGET]       = { .type = NLA_NESTED, .type_system = &rtnl_bond_arp_type_system }, -        [IFLA_BOND_ARP_VALIDATE]        = { .type = NLA_U32 }, -        [IFLA_BOND_ARP_ALL_TARGETS]     = { .type = NLA_U32 }, -        [IFLA_BOND_PRIMARY]             = { .type = NLA_U32 }, -        [IFLA_BOND_PRIMARY_RESELECT]    = { .type = NLA_U8 }, -        [IFLA_BOND_FAIL_OVER_MAC]       = { .type = NLA_U8 }, -        [IFLA_BOND_XMIT_HASH_POLICY]    = { .type = NLA_U8 }, -        [IFLA_BOND_RESEND_IGMP]         = { .type = NLA_U32 }, -        [IFLA_BOND_NUM_PEER_NOTIF]      = { .type = NLA_U8 }, -        [IFLA_BOND_ALL_SLAVES_ACTIVE]   = { .type = NLA_U8 }, -        [IFLA_BOND_MIN_LINKS]           = { .type = NLA_U32 }, -        [IFLA_BOND_LP_INTERVAL]         = { .type = NLA_U32 }, -        [IFLA_BOND_PACKETS_PER_SLAVE]   = { .type = NLA_U32 }, -        [IFLA_BOND_AD_LACP_RATE]        = { .type = NLA_U8 }, -        [IFLA_BOND_AD_SELECT]           = { .type = NLA_U8 }, -        [IFLA_BOND_AD_INFO]             = { .type = NLA_NESTED }, +        [IFLA_BOND_MODE]                = { .type = NETLINK_TYPE_U8 }, +        [IFLA_BOND_ACTIVE_SLAVE]        = { .type = NETLINK_TYPE_U32 }, +        [IFLA_BOND_MIIMON]              = { .type = NETLINK_TYPE_U32 }, +        [IFLA_BOND_UPDELAY]             = { .type = NETLINK_TYPE_U32 }, +        [IFLA_BOND_DOWNDELAY]           = { .type = NETLINK_TYPE_U32 }, +        [IFLA_BOND_USE_CARRIER]         = { .type = NETLINK_TYPE_U8 }, +        [IFLA_BOND_ARP_INTERVAL]        = { .type = NETLINK_TYPE_U32 }, +        [IFLA_BOND_ARP_IP_TARGET]       = { .type = NETLINK_TYPE_NESTED, .type_system = &rtnl_bond_arp_type_system }, +        [IFLA_BOND_ARP_VALIDATE]        = { .type = NETLINK_TYPE_U32 }, +        [IFLA_BOND_ARP_ALL_TARGETS]     = { .type = NETLINK_TYPE_U32 }, +        [IFLA_BOND_PRIMARY]             = { .type = NETLINK_TYPE_U32 }, +        [IFLA_BOND_PRIMARY_RESELECT]    = { .type = NETLINK_TYPE_U8 }, +        [IFLA_BOND_FAIL_OVER_MAC]       = { .type = NETLINK_TYPE_U8 }, +        [IFLA_BOND_XMIT_HASH_POLICY]    = { .type = NETLINK_TYPE_U8 }, +        [IFLA_BOND_RESEND_IGMP]         = { .type = NETLINK_TYPE_U32 }, +        [IFLA_BOND_NUM_PEER_NOTIF]      = { .type = NETLINK_TYPE_U8 }, +        [IFLA_BOND_ALL_SLAVES_ACTIVE]   = { .type = NETLINK_TYPE_U8 }, +        [IFLA_BOND_MIN_LINKS]           = { .type = NETLINK_TYPE_U32 }, +        [IFLA_BOND_LP_INTERVAL]         = { .type = NETLINK_TYPE_U32 }, +        [IFLA_BOND_PACKETS_PER_SLAVE]   = { .type = NETLINK_TYPE_U32 }, +        [IFLA_BOND_AD_LACP_RATE]        = { .type = NETLINK_TYPE_U8 }, +        [IFLA_BOND_AD_SELECT]           = { .type = NETLINK_TYPE_U8 }, +        [IFLA_BOND_AD_INFO]             = { .type = NETLINK_TYPE_NESTED },  };  static const NLType rtnl_link_info_data_iptun_types[IFLA_IPTUN_MAX + 1] = { -        [IFLA_IPTUN_LINK]                = { .type = NLA_U32 }, -        [IFLA_IPTUN_LOCAL]               = { .type = NLA_IN_ADDR }, -        [IFLA_IPTUN_REMOTE]              = { .type = NLA_IN_ADDR }, -        [IFLA_IPTUN_TTL]                 = { .type = NLA_U8 }, -        [IFLA_IPTUN_TOS]                 = { .type = NLA_U8 }, -        [IFLA_IPTUN_PMTUDISC]            = { .type = NLA_U8 }, -        [IFLA_IPTUN_FLAGS]               = { .type = NLA_U16 }, -        [IFLA_IPTUN_PROTO]               = { .type = NLA_U8 }, -        [IFLA_IPTUN_6RD_PREFIX]          = { .type = NLA_IN_ADDR }, -        [IFLA_IPTUN_6RD_RELAY_PREFIX]    = { .type = NLA_U32 }, -        [IFLA_IPTUN_6RD_PREFIXLEN]       = { .type = NLA_U16 }, -        [IFLA_IPTUN_6RD_RELAY_PREFIXLEN] = { .type = NLA_U16 }, +        [IFLA_IPTUN_LINK]                = { .type = NETLINK_TYPE_U32 }, +        [IFLA_IPTUN_LOCAL]               = { .type = NETLINK_TYPE_IN_ADDR }, +        [IFLA_IPTUN_REMOTE]              = { .type = NETLINK_TYPE_IN_ADDR }, +        [IFLA_IPTUN_TTL]                 = { .type = NETLINK_TYPE_U8 }, +        [IFLA_IPTUN_TOS]                 = { .type = NETLINK_TYPE_U8 }, +        [IFLA_IPTUN_PMTUDISC]            = { .type = NETLINK_TYPE_U8 }, +        [IFLA_IPTUN_FLAGS]               = { .type = NETLINK_TYPE_U16 }, +        [IFLA_IPTUN_PROTO]               = { .type = NETLINK_TYPE_U8 }, +        [IFLA_IPTUN_6RD_PREFIX]          = { .type = NETLINK_TYPE_IN_ADDR }, +        [IFLA_IPTUN_6RD_RELAY_PREFIX]    = { .type = NETLINK_TYPE_U32 }, +        [IFLA_IPTUN_6RD_PREFIXLEN]       = { .type = NETLINK_TYPE_U16 }, +        [IFLA_IPTUN_6RD_RELAY_PREFIXLEN] = { .type = NETLINK_TYPE_U16 }, +        [IFLA_IPTUN_ENCAP_TYPE]          = { .type = NETLINK_TYPE_U16 }, +        [IFLA_IPTUN_ENCAP_FLAGS]         = { .type = NETLINK_TYPE_U16 }, +        [IFLA_IPTUN_ENCAP_SPORT]         = { .type = NETLINK_TYPE_U16 }, +        [IFLA_IPTUN_ENCAP_DPORT]         = { .type = NETLINK_TYPE_U16 },  };  static  const NLType rtnl_link_info_data_ipgre_types[IFLA_GRE_MAX + 1] = { -        [IFLA_GRE_LINK]     = { .type = NLA_U32 }, -        [IFLA_GRE_IFLAGS]   = { .type = NLA_U16 }, -        [IFLA_GRE_OFLAGS]   = { .type = NLA_U16 }, -        [IFLA_GRE_IKEY]     = { .type = NLA_U32 }, -        [IFLA_GRE_OKEY]     = { .type = NLA_U32 }, -        [IFLA_GRE_LOCAL]    = { .type = NLA_IN_ADDR }, -        [IFLA_GRE_REMOTE]   = { .type = NLA_IN_ADDR }, -        [IFLA_GRE_TTL]      = { .type = NLA_U8 }, -        [IFLA_GRE_TOS]      = { .type = NLA_U8 }, -        [IFLA_GRE_PMTUDISC] = { .type = NLA_U8 }, +        [IFLA_GRE_LINK]         = { .type = NETLINK_TYPE_U32 }, +        [IFLA_GRE_IFLAGS]       = { .type = NETLINK_TYPE_U16 }, +        [IFLA_GRE_OFLAGS]       = { .type = NETLINK_TYPE_U16 }, +        [IFLA_GRE_IKEY]         = { .type = NETLINK_TYPE_U32 }, +        [IFLA_GRE_OKEY]         = { .type = NETLINK_TYPE_U32 }, +        [IFLA_GRE_LOCAL]        = { .type = NETLINK_TYPE_IN_ADDR }, +        [IFLA_GRE_REMOTE]       = { .type = NETLINK_TYPE_IN_ADDR }, +        [IFLA_GRE_TTL]          = { .type = NETLINK_TYPE_U8 }, +        [IFLA_GRE_TOS]          = { .type = NETLINK_TYPE_U8 }, +        [IFLA_GRE_PMTUDISC]     = { .type = NETLINK_TYPE_U8 }, +        [IFLA_GRE_FLOWINFO]     = { .type = NETLINK_TYPE_U32 }, +        [IFLA_GRE_FLAGS]        = { .type = NETLINK_TYPE_U32 }, +        [IFLA_GRE_ENCAP_TYPE]   = { .type = NETLINK_TYPE_U16 }, +        [IFLA_GRE_ENCAP_FLAGS]  = { .type = NETLINK_TYPE_U16 }, +        [IFLA_GRE_ENCAP_SPORT]  = { .type = NETLINK_TYPE_U16 }, +        [IFLA_GRE_ENCAP_DPORT]  = { .type = NETLINK_TYPE_U16 },  };  static const NLType rtnl_link_info_data_ipvti_types[IFLA_VTI_MAX + 1] = { -        [IFLA_VTI_LINK]         = { .type = NLA_U32 }, -        [IFLA_VTI_IKEY]         = { .type = NLA_U32 }, -        [IFLA_VTI_OKEY]         = { .type = NLA_U32 }, -        [IFLA_VTI_LOCAL]        = { .type = NLA_IN_ADDR  }, -        [IFLA_VTI_REMOTE]       = { .type = NLA_IN_ADDR  }, +        [IFLA_VTI_LINK]         = { .type = NETLINK_TYPE_U32 }, +        [IFLA_VTI_IKEY]         = { .type = NETLINK_TYPE_U32 }, +        [IFLA_VTI_OKEY]         = { .type = NETLINK_TYPE_U32 }, +        [IFLA_VTI_LOCAL]        = { .type = NETLINK_TYPE_IN_ADDR }, +        [IFLA_VTI_REMOTE]       = { .type = NETLINK_TYPE_IN_ADDR },  };  static const NLType rtnl_link_info_data_ip6tnl_types[IFLA_IPTUN_MAX + 1] = { -        [IFLA_IPTUN_LINK]                = { .type = NLA_U32 }, -        [IFLA_IPTUN_LOCAL]               = { .type = NLA_IN_ADDR }, -        [IFLA_IPTUN_REMOTE]              = { .type = NLA_IN_ADDR }, -        [IFLA_IPTUN_TTL]                 = { .type = NLA_U8 }, -        [IFLA_IPTUN_FLAGS]               = { .type = NLA_U32 }, -        [IFLA_IPTUN_PROTO]               = { .type = NLA_U8 }, -        [IFLA_IPTUN_ENCAP_LIMIT]         = { .type = NLA_U8 }, -        [IFLA_IPTUN_FLOWINFO]            = { .type = NLA_U32}, +        [IFLA_IPTUN_LINK]                = { .type = NETLINK_TYPE_U32 }, +        [IFLA_IPTUN_LOCAL]               = { .type = NETLINK_TYPE_IN_ADDR }, +        [IFLA_IPTUN_REMOTE]              = { .type = NETLINK_TYPE_IN_ADDR }, +        [IFLA_IPTUN_TTL]                 = { .type = NETLINK_TYPE_U8 }, +        [IFLA_IPTUN_FLAGS]               = { .type = NETLINK_TYPE_U32 }, +        [IFLA_IPTUN_PROTO]               = { .type = NETLINK_TYPE_U8 }, +        [IFLA_IPTUN_ENCAP_LIMIT]         = { .type = NETLINK_TYPE_U8 }, +        [IFLA_IPTUN_FLOWINFO]            = { .type = NETLINK_TYPE_U32 },  };  /* these strings must match the .kind entries in the kernel */ @@ -211,37 +264,37 @@ static const char* const nl_union_link_info_data_table[_NL_UNION_LINK_INFO_DATA_  DEFINE_STRING_TABLE_LOOKUP(nl_union_link_info_data, NLUnionLinkInfoData);  static const NLTypeSystem rtnl_link_info_data_type_systems[_NL_UNION_LINK_INFO_DATA_MAX] = { -        [NL_UNION_LINK_INFO_DATA_BOND] =        { .max = ELEMENTSOF(rtnl_link_info_data_bond_types) - 1, +        [NL_UNION_LINK_INFO_DATA_BOND] =        { .count = ELEMENTSOF(rtnl_link_info_data_bond_types),                                                    .types = rtnl_link_info_data_bond_types }, -        [NL_UNION_LINK_INFO_DATA_BRIDGE] =      { .max = ELEMENTSOF(rtnl_link_info_data_bridge_types) - 1, +        [NL_UNION_LINK_INFO_DATA_BRIDGE] =      { .count = ELEMENTSOF(rtnl_link_info_data_bridge_types),                                                    .types = rtnl_link_info_data_bridge_types }, -        [NL_UNION_LINK_INFO_DATA_VLAN] =        { .max = ELEMENTSOF(rtnl_link_info_data_vlan_types) - 1, +        [NL_UNION_LINK_INFO_DATA_VLAN] =        { .count = ELEMENTSOF(rtnl_link_info_data_vlan_types),                                                    .types = rtnl_link_info_data_vlan_types }, -        [NL_UNION_LINK_INFO_DATA_VETH] =        { .max = ELEMENTSOF(rtnl_link_info_data_veth_types) - 1, +        [NL_UNION_LINK_INFO_DATA_VETH] =        { .count = ELEMENTSOF(rtnl_link_info_data_veth_types),                                                    .types = rtnl_link_info_data_veth_types }, -        [NL_UNION_LINK_INFO_DATA_MACVLAN] =     { .max = ELEMENTSOF(rtnl_link_info_data_macvlan_types) - 1, +        [NL_UNION_LINK_INFO_DATA_MACVLAN] =     { .count = ELEMENTSOF(rtnl_link_info_data_macvlan_types),                                                    .types = rtnl_link_info_data_macvlan_types }, -        [NL_UNION_LINK_INFO_DATA_IPVLAN] =      { .max = ELEMENTSOF(rtnl_link_info_data_ipvlan_types) - 1, +        [NL_UNION_LINK_INFO_DATA_IPVLAN] =      { .count = ELEMENTSOF(rtnl_link_info_data_ipvlan_types),                                                    .types = rtnl_link_info_data_ipvlan_types }, -        [NL_UNION_LINK_INFO_DATA_VXLAN] =       { .max = ELEMENTSOF(rtnl_link_info_data_vxlan_types) - 1, +        [NL_UNION_LINK_INFO_DATA_VXLAN] =       { .count = ELEMENTSOF(rtnl_link_info_data_vxlan_types),                                                    .types = rtnl_link_info_data_vxlan_types }, -        [NL_UNION_LINK_INFO_DATA_IPIP_TUNNEL] = { .max = ELEMENTSOF(rtnl_link_info_data_iptun_types) - 1, +        [NL_UNION_LINK_INFO_DATA_IPIP_TUNNEL] = { .count = ELEMENTSOF(rtnl_link_info_data_iptun_types),                                                    .types = rtnl_link_info_data_iptun_types }, -        [NL_UNION_LINK_INFO_DATA_IPGRE_TUNNEL] =  { .max = ELEMENTSOF(rtnl_link_info_data_ipgre_types) - 1, +        [NL_UNION_LINK_INFO_DATA_IPGRE_TUNNEL] =  { .count = ELEMENTSOF(rtnl_link_info_data_ipgre_types),                                                      .types = rtnl_link_info_data_ipgre_types }, -        [NL_UNION_LINK_INFO_DATA_IPGRETAP_TUNNEL] =  { .max = ELEMENTSOF(rtnl_link_info_data_ipgre_types) - 1, +        [NL_UNION_LINK_INFO_DATA_IPGRETAP_TUNNEL] =  { .count = ELEMENTSOF(rtnl_link_info_data_ipgre_types),                                                      .types = rtnl_link_info_data_ipgre_types }, -        [NL_UNION_LINK_INFO_DATA_IP6GRE_TUNNEL] =  { .max = ELEMENTSOF(rtnl_link_info_data_ipgre_types) - 1, +        [NL_UNION_LINK_INFO_DATA_IP6GRE_TUNNEL] =  { .count = ELEMENTSOF(rtnl_link_info_data_ipgre_types),                                                      .types = rtnl_link_info_data_ipgre_types }, -        [NL_UNION_LINK_INFO_DATA_IP6GRETAP_TUNNEL] =  { .max = ELEMENTSOF(rtnl_link_info_data_ipgre_types) - 1, +        [NL_UNION_LINK_INFO_DATA_IP6GRETAP_TUNNEL] =  { .count = ELEMENTSOF(rtnl_link_info_data_ipgre_types),                                                      .types = rtnl_link_info_data_ipgre_types }, -        [NL_UNION_LINK_INFO_DATA_SIT_TUNNEL] =  { .max = ELEMENTSOF(rtnl_link_info_data_iptun_types) - 1, +        [NL_UNION_LINK_INFO_DATA_SIT_TUNNEL] =  { .count = ELEMENTSOF(rtnl_link_info_data_iptun_types),                                                    .types = rtnl_link_info_data_iptun_types }, -        [NL_UNION_LINK_INFO_DATA_VTI_TUNNEL] =  { .max = ELEMENTSOF(rtnl_link_info_data_ipvti_types) - 1, +        [NL_UNION_LINK_INFO_DATA_VTI_TUNNEL] =  { .count = ELEMENTSOF(rtnl_link_info_data_ipvti_types),                                                    .types = rtnl_link_info_data_ipvti_types }, -        [NL_UNION_LINK_INFO_DATA_VTI6_TUNNEL] =  { .max = ELEMENTSOF(rtnl_link_info_data_ipvti_types) - 1, +        [NL_UNION_LINK_INFO_DATA_VTI6_TUNNEL] =  { .count = ELEMENTSOF(rtnl_link_info_data_ipvti_types),                                                    .types = rtnl_link_info_data_ipvti_types }, -        [NL_UNION_LINK_INFO_DATA_IP6TNL_TUNNEL] =  { .max = ELEMENTSOF(rtnl_link_info_data_ip6tnl_types) - 1, +        [NL_UNION_LINK_INFO_DATA_IP6TNL_TUNNEL] =  { .count = ELEMENTSOF(rtnl_link_info_data_ip6tnl_types),                                                       .types = rtnl_link_info_data_ip6tnl_types },  }; @@ -255,33 +308,36 @@ static const NLTypeSystemUnion rtnl_link_info_data_type_system_union = {  };  static const NLType rtnl_link_info_types[IFLA_INFO_MAX + 1] = { -        [IFLA_INFO_KIND]        = { .type = NLA_STRING }, -        [IFLA_INFO_DATA]        = { .type = NLA_UNION, .type_system_union = &rtnl_link_info_data_type_system_union}, +        [IFLA_INFO_KIND]        = { .type = NETLINK_TYPE_STRING }, +        [IFLA_INFO_DATA]        = { .type = NETLINK_TYPE_UNION, .type_system_union = &rtnl_link_info_data_type_system_union},  /*          [IFLA_INFO_XSTATS], -        [IFLA_INFO_SLAVE_KIND]  = { .type = NLA_STRING }, -        [IFLA_INFO_SLAVE_DATA]  = { .type = NLA_NESTED }, +        [IFLA_INFO_SLAVE_KIND]  = { .type = NETLINK_TYPE_STRING }, +        [IFLA_INFO_SLAVE_DATA]  = { .type = NETLINK_TYPE_NESTED },  */  };  static const NLTypeSystem rtnl_link_info_type_system = { -        .max = ELEMENTSOF(rtnl_link_info_types) - 1, +        .count = ELEMENTSOF(rtnl_link_info_types),          .types = rtnl_link_info_types,  };  static const struct NLType rtnl_prot_info_bridge_port_types[IFLA_BRPORT_MAX + 1] = { -        [IFLA_BRPORT_STATE]     = { .type = NLA_U8 }, -        [IFLA_BRPORT_COST]      = { .type = NLA_U32 }, -        [IFLA_BRPORT_PRIORITY]  = { .type = NLA_U16 }, -        [IFLA_BRPORT_MODE]      = { .type = NLA_U8 }, -        [IFLA_BRPORT_GUARD]     = { .type = NLA_U8 }, -        [IFLA_BRPORT_PROTECT]   = { .type = NLA_U8 }, -        [IFLA_BRPORT_LEARNING]  = { .type = NLA_U8 }, -        [IFLA_BRPORT_UNICAST_FLOOD] = { .type = NLA_U8 }, +        [IFLA_BRPORT_STATE]             = { .type = NETLINK_TYPE_U8 }, +        [IFLA_BRPORT_COST]              = { .type = NETLINK_TYPE_U32 }, +        [IFLA_BRPORT_PRIORITY]          = { .type = NETLINK_TYPE_U16 }, +        [IFLA_BRPORT_MODE]              = { .type = NETLINK_TYPE_U8 }, +        [IFLA_BRPORT_GUARD]             = { .type = NETLINK_TYPE_U8 }, +        [IFLA_BRPORT_PROTECT]           = { .type = NETLINK_TYPE_U8 }, +        [IFLA_BRPORT_FAST_LEAVE]        = { .type = NETLINK_TYPE_U8 }, +        [IFLA_BRPORT_LEARNING]          = { .type = NETLINK_TYPE_U8 }, +        [IFLA_BRPORT_UNICAST_FLOOD]     = { .type = NETLINK_TYPE_U8 }, +        [IFLA_BRPORT_PROXYARP]          = { .type = NETLINK_TYPE_U8 }, +        [IFLA_BRPORT_LEARNING_SYNC]     = { .type = NETLINK_TYPE_U8 },  };  static const NLTypeSystem rtnl_prot_info_type_systems[AF_MAX] = { -        [AF_BRIDGE] =   { .max = ELEMENTSOF(rtnl_prot_info_bridge_port_types) - 1, +        [AF_BRIDGE] =   { .count = ELEMENTSOF(rtnl_prot_info_bridge_port_types),                            .types = rtnl_prot_info_bridge_port_types },  }; @@ -292,7 +348,7 @@ static const NLTypeSystemUnion rtnl_prot_info_type_system_union = {  };  static const struct NLType rtnl_af_spec_inet6_types[IFLA_INET6_MAX + 1] = { -        [IFLA_INET6_FLAGS]              = { .type = NLA_U32 }, +        [IFLA_INET6_FLAGS]              = { .type = NETLINK_TYPE_U32 },  /*          IFLA_INET6_CONF,          IFLA_INET6_STATS, @@ -300,114 +356,114 @@ static const struct NLType rtnl_af_spec_inet6_types[IFLA_INET6_MAX + 1] = {          IFLA_INET6_CACHEINFO,          IFLA_INET6_ICMP6STATS,  */ -        [IFLA_INET6_TOKEN]              = { .type = NLA_IN_ADDR }, -        [IFLA_INET6_ADDR_GEN_MODE]      = { .type = NLA_U8 }, +        [IFLA_INET6_TOKEN]              = { .type = NETLINK_TYPE_IN_ADDR }, +        [IFLA_INET6_ADDR_GEN_MODE]      = { .type = NETLINK_TYPE_U8 },  };  static const NLTypeSystem rtnl_af_spec_inet6_type_system = { -        .max = ELEMENTSOF(rtnl_af_spec_inet6_types) - 1, +        .count = ELEMENTSOF(rtnl_af_spec_inet6_types),          .types = rtnl_af_spec_inet6_types,  };  static const NLType rtnl_af_spec_types[AF_MAX + 1] = { -        [AF_INET6] =    { .type = NLA_NESTED, .type_system = &rtnl_af_spec_inet6_type_system }, +        [AF_INET6] =    { .type = NETLINK_TYPE_NESTED, .type_system = &rtnl_af_spec_inet6_type_system },  };  static const NLTypeSystem rtnl_af_spec_type_system = { -        .max = ELEMENTSOF(rtnl_af_spec_types) - 1, +        .count = ELEMENTSOF(rtnl_af_spec_types),          .types = rtnl_af_spec_types,  };  static const NLType rtnl_link_types[IFLA_MAX + 1 ] = { -        [IFLA_ADDRESS]          = { .type = NLA_ETHER_ADDR, }, -        [IFLA_BROADCAST]        = { .type = NLA_ETHER_ADDR, }, -        [IFLA_IFNAME]           = { .type = NLA_STRING, .size = IFNAMSIZ - 1, }, -        [IFLA_MTU]              = { .type = NLA_U32 }, -        [IFLA_LINK]             = { .type = NLA_U32 }, +        [IFLA_ADDRESS]          = { .type = NETLINK_TYPE_ETHER_ADDR }, +        [IFLA_BROADCAST]        = { .type = NETLINK_TYPE_ETHER_ADDR }, +        [IFLA_IFNAME]           = { .type = NETLINK_TYPE_STRING, .size = IFNAMSIZ - 1 }, +        [IFLA_MTU]              = { .type = NETLINK_TYPE_U32 }, +        [IFLA_LINK]             = { .type = NETLINK_TYPE_U32 },  /*          [IFLA_QDISC],          [IFLA_STATS],          [IFLA_COST],          [IFLA_PRIORITY],  */ -        [IFLA_MASTER]           = { .type = NLA_U32 }, +        [IFLA_MASTER]           = { .type = NETLINK_TYPE_U32 },  /*          [IFLA_WIRELESS],  */ -        [IFLA_PROTINFO]         = { .type = NLA_UNION, .type_system_union = &rtnl_prot_info_type_system_union }, -        [IFLA_TXQLEN]           = { .type = NLA_U32 }, +        [IFLA_PROTINFO]         = { .type = NETLINK_TYPE_UNION, .type_system_union = &rtnl_prot_info_type_system_union }, +        [IFLA_TXQLEN]           = { .type = NETLINK_TYPE_U32 },  /*          [IFLA_MAP]              = { .len = sizeof(struct rtnl_link_ifmap) },  */ -        [IFLA_WEIGHT]           = { .type = NLA_U32 }, -        [IFLA_OPERSTATE]        = { .type = NLA_U8 }, -        [IFLA_LINKMODE]         = { .type = NLA_U8 }, -        [IFLA_LINKINFO]         = { .type = NLA_NESTED, .type_system = &rtnl_link_info_type_system }, -        [IFLA_NET_NS_PID]       = { .type = NLA_U32 }, -        [IFLA_IFALIAS]          = { .type = NLA_STRING, .size = IFALIASZ - 1 }, +        [IFLA_WEIGHT]           = { .type = NETLINK_TYPE_U32 }, +        [IFLA_OPERSTATE]        = { .type = NETLINK_TYPE_U8 }, +        [IFLA_LINKMODE]         = { .type = NETLINK_TYPE_U8 }, +        [IFLA_LINKINFO]         = { .type = NETLINK_TYPE_NESTED, .type_system = &rtnl_link_info_type_system }, +        [IFLA_NET_NS_PID]       = { .type = NETLINK_TYPE_U32 }, +        [IFLA_IFALIAS]          = { .type = NETLINK_TYPE_STRING, .size = IFALIASZ - 1 },  /*          [IFLA_NUM_VF], -        [IFLA_VFINFO_LIST]      = {. type = NLA_NESTED, }, +        [IFLA_VFINFO_LIST]      = {. type = NETLINK_TYPE_NESTED, },          [IFLA_STATS64], -        [IFLA_VF_PORTS]         = { .type = NLA_NESTED }, -        [IFLA_PORT_SELF]        = { .type = NLA_NESTED }, +        [IFLA_VF_PORTS]         = { .type = NETLINK_TYPE_NESTED }, +        [IFLA_PORT_SELF]        = { .type = NETLINK_TYPE_NESTED },  */ -        [IFLA_AF_SPEC]          = { .type = NLA_NESTED, .type_system = &rtnl_af_spec_type_system }, +        [IFLA_AF_SPEC]          = { .type = NETLINK_TYPE_NESTED, .type_system = &rtnl_af_spec_type_system },  /*          [IFLA_VF_PORTS],          [IFLA_PORT_SELF],          [IFLA_AF_SPEC],  */ -        [IFLA_GROUP]            = { .type = NLA_U32 }, -        [IFLA_NET_NS_FD]        = { .type = NLA_U32 }, -        [IFLA_EXT_MASK]         = { .type = NLA_U32 }, -        [IFLA_PROMISCUITY]      = { .type = NLA_U32 }, -        [IFLA_NUM_TX_QUEUES]    = { .type = NLA_U32 }, -        [IFLA_NUM_RX_QUEUES]    = { .type = NLA_U32 }, -        [IFLA_CARRIER]          = { .type = NLA_U8 }, +        [IFLA_GROUP]            = { .type = NETLINK_TYPE_U32 }, +        [IFLA_NET_NS_FD]        = { .type = NETLINK_TYPE_U32 }, +        [IFLA_EXT_MASK]         = { .type = NETLINK_TYPE_U32 }, +        [IFLA_PROMISCUITY]      = { .type = NETLINK_TYPE_U32 }, +        [IFLA_NUM_TX_QUEUES]    = { .type = NETLINK_TYPE_U32 }, +        [IFLA_NUM_RX_QUEUES]    = { .type = NETLINK_TYPE_U32 }, +        [IFLA_CARRIER]          = { .type = NETLINK_TYPE_U8 },  /* -        [IFLA_PHYS_PORT_ID]     = { .type = NLA_BINARY, .len = MAX_PHYS_PORT_ID_LEN }, +        [IFLA_PHYS_PORT_ID]     = { .type = NETLINK_TYPE_BINARY, .len = MAX_PHYS_PORT_ID_LEN },  */  };  static const NLTypeSystem rtnl_link_type_system = { -        .max = ELEMENTSOF(rtnl_link_types) - 1, +        .count = ELEMENTSOF(rtnl_link_types),          .types = rtnl_link_types,  };  /* IFA_FLAGS was defined in kernel 3.14, but we still support older   * kernels where IFA_MAX is lower. */  static const NLType rtnl_address_types[CONST_MAX(IFA_MAX, IFA_FLAGS) + 1] = { -        [IFA_ADDRESS]           = { .type = NLA_IN_ADDR }, -        [IFA_LOCAL]             = { .type = NLA_IN_ADDR }, -        [IFA_LABEL]             = { .type = NLA_STRING, .size = IFNAMSIZ - 1 }, -        [IFA_BROADCAST]         = { .type = NLA_IN_ADDR }, /* 6? */ -        [IFA_CACHEINFO]         = { .type = NLA_CACHE_INFO, .size = sizeof(struct ifa_cacheinfo) }, +        [IFA_ADDRESS]           = { .type = NETLINK_TYPE_IN_ADDR }, +        [IFA_LOCAL]             = { .type = NETLINK_TYPE_IN_ADDR }, +        [IFA_LABEL]             = { .type = NETLINK_TYPE_STRING, .size = IFNAMSIZ - 1 }, +        [IFA_BROADCAST]         = { .type = NETLINK_TYPE_IN_ADDR }, /* 6? */ +        [IFA_CACHEINFO]         = { .type = NETLINK_TYPE_CACHE_INFO, .size = sizeof(struct ifa_cacheinfo) },  /*          [IFA_ANYCAST],          [IFA_MULTICAST],  */ -        [IFA_FLAGS]             = { .type = NLA_U32 }, +        [IFA_FLAGS]             = { .type = NETLINK_TYPE_U32 },  };  static const NLTypeSystem rtnl_address_type_system = { -        .max = ELEMENTSOF(rtnl_address_types) - 1, +        .count = ELEMENTSOF(rtnl_address_types),          .types = rtnl_address_types,  };  static const NLType rtnl_route_types[RTA_MAX + 1] = { -        [RTA_DST]               = { .type = NLA_IN_ADDR }, /* 6? */ -        [RTA_SRC]               = { .type = NLA_IN_ADDR }, /* 6? */ -        [RTA_IIF]               = { .type = NLA_U32 }, -        [RTA_OIF]               = { .type = NLA_U32 }, -        [RTA_GATEWAY]           = { .type = NLA_IN_ADDR }, -        [RTA_PRIORITY]          = { .type = NLA_U32 }, -        [RTA_PREFSRC]           = { .type = NLA_IN_ADDR }, /* 6? */ +        [RTA_DST]               = { .type = NETLINK_TYPE_IN_ADDR }, /* 6? */ +        [RTA_SRC]               = { .type = NETLINK_TYPE_IN_ADDR }, /* 6? */ +        [RTA_IIF]               = { .type = NETLINK_TYPE_U32 }, +        [RTA_OIF]               = { .type = NETLINK_TYPE_U32 }, +        [RTA_GATEWAY]           = { .type = NETLINK_TYPE_IN_ADDR }, +        [RTA_PRIORITY]          = { .type = NETLINK_TYPE_U32 }, +        [RTA_PREFSRC]           = { .type = NETLINK_TYPE_IN_ADDR }, /* 6? */  /* -        [RTA_METRICS]           = { .type = NLA_NESTED }, +        [RTA_METRICS]           = { .type = NETLINK_TYPE_NESTED },          [RTA_MULTIPATH]         = { .len = sizeof(struct rtnexthop) },  */ -        [RTA_FLOW]              = { .type = NLA_U32 }, /* 6? */ +        [RTA_FLOW]              = { .type = NETLINK_TYPE_U32 }, /* 6? */  /*          RTA_CACHEINFO,          RTA_TABLE, @@ -417,65 +473,95 @@ static const NLType rtnl_route_types[RTA_MAX + 1] = {  };  static const NLTypeSystem rtnl_route_type_system = { -        .max = ELEMENTSOF(rtnl_route_types) - 1, +        .count = ELEMENTSOF(rtnl_route_types),          .types = rtnl_route_types,  };  static const NLType rtnl_neigh_types[NDA_MAX + 1] = { -        [NDA_DST]               = { .type = NLA_IN_ADDR }, -        [NDA_LLADDR]            = { .type = NLA_ETHER_ADDR }, -        [NDA_CACHEINFO]         = { .type = NLA_CACHE_INFO, .size = sizeof(struct nda_cacheinfo) }, -        [NDA_PROBES]            = { .type = NLA_U32 }, -        [NDA_VLAN]              = { .type = NLA_U16 }, -        [NDA_PORT]              = { .type = NLA_U16 }, -        [NDA_VNI]               = { .type = NLA_U32 }, -        [NDA_IFINDEX]           = { .type = NLA_U32 }, +        [NDA_DST]               = { .type = NETLINK_TYPE_IN_ADDR }, +        [NDA_LLADDR]            = { .type = NETLINK_TYPE_ETHER_ADDR }, +        [NDA_CACHEINFO]         = { .type = NETLINK_TYPE_CACHE_INFO, .size = sizeof(struct nda_cacheinfo) }, +        [NDA_PROBES]            = { .type = NETLINK_TYPE_U32 }, +        [NDA_VLAN]              = { .type = NETLINK_TYPE_U16 }, +        [NDA_PORT]              = { .type = NETLINK_TYPE_U16 }, +        [NDA_VNI]               = { .type = NETLINK_TYPE_U32 }, +        [NDA_IFINDEX]           = { .type = NETLINK_TYPE_U32 },  };  static const NLTypeSystem rtnl_neigh_type_system = { -        .max = ELEMENTSOF(rtnl_neigh_types) - 1, +        .count = ELEMENTSOF(rtnl_neigh_types),          .types = rtnl_neigh_types,  };  static const NLType rtnl_types[RTM_MAX + 1] = { -        [NLMSG_DONE]   = { .type = NLA_META, .size = 0 }, -        [NLMSG_ERROR]  = { .type = NLA_META, .size = sizeof(struct nlmsgerr) }, -        [RTM_NEWLINK]  = { .type = NLA_NESTED, .type_system = &rtnl_link_type_system, .size = sizeof(struct ifinfomsg) }, -        [RTM_DELLINK]  = { .type = NLA_NESTED, .type_system = &rtnl_link_type_system, .size = sizeof(struct ifinfomsg) }, -        [RTM_GETLINK]  = { .type = NLA_NESTED, .type_system = &rtnl_link_type_system, .size = sizeof(struct ifinfomsg) }, -        [RTM_SETLINK]  = { .type = NLA_NESTED, .type_system = &rtnl_link_type_system, .size = sizeof(struct ifinfomsg) }, -        [RTM_NEWADDR]  = { .type = NLA_NESTED, .type_system = &rtnl_address_type_system, .size = sizeof(struct ifaddrmsg) }, -        [RTM_DELADDR]  = { .type = NLA_NESTED, .type_system = &rtnl_address_type_system, .size = sizeof(struct ifaddrmsg) }, -        [RTM_GETADDR]  = { .type = NLA_NESTED, .type_system = &rtnl_address_type_system, .size = sizeof(struct ifaddrmsg) }, -        [RTM_NEWROUTE] = { .type = NLA_NESTED, .type_system = &rtnl_route_type_system, .size = sizeof(struct rtmsg) }, -        [RTM_DELROUTE] = { .type = NLA_NESTED, .type_system = &rtnl_route_type_system, .size = sizeof(struct rtmsg) }, -        [RTM_GETROUTE] = { .type = NLA_NESTED, .type_system = &rtnl_route_type_system, .size = sizeof(struct rtmsg) }, -        [RTM_NEWNEIGH] = { .type = NLA_NESTED, .type_system = &rtnl_neigh_type_system, .size = sizeof(struct ndmsg) }, -        [RTM_DELNEIGH] = { .type = NLA_NESTED, .type_system = &rtnl_neigh_type_system, .size = sizeof(struct ndmsg) }, -        [RTM_GETNEIGH] = { .type = NLA_NESTED, .type_system = &rtnl_neigh_type_system, .size = sizeof(struct ndmsg) }, -}; - -const NLTypeSystem rtnl_type_system = { -        .max = ELEMENTSOF(rtnl_types) - 1, +        [NLMSG_DONE]   = { .type = NETLINK_TYPE_NESTED, .type_system = &empty_type_system, .size = 0 }, +        [NLMSG_ERROR]  = { .type = NETLINK_TYPE_NESTED, .type_system = &empty_type_system, .size = sizeof(struct nlmsgerr) }, +        [RTM_NEWLINK]  = { .type = NETLINK_TYPE_NESTED, .type_system = &rtnl_link_type_system, .size = sizeof(struct ifinfomsg) }, +        [RTM_DELLINK]  = { .type = NETLINK_TYPE_NESTED, .type_system = &rtnl_link_type_system, .size = sizeof(struct ifinfomsg) }, +        [RTM_GETLINK]  = { .type = NETLINK_TYPE_NESTED, .type_system = &rtnl_link_type_system, .size = sizeof(struct ifinfomsg) }, +        [RTM_SETLINK]  = { .type = NETLINK_TYPE_NESTED, .type_system = &rtnl_link_type_system, .size = sizeof(struct ifinfomsg) }, +        [RTM_NEWADDR]  = { .type = NETLINK_TYPE_NESTED, .type_system = &rtnl_address_type_system, .size = sizeof(struct ifaddrmsg) }, +        [RTM_DELADDR]  = { .type = NETLINK_TYPE_NESTED, .type_system = &rtnl_address_type_system, .size = sizeof(struct ifaddrmsg) }, +        [RTM_GETADDR]  = { .type = NETLINK_TYPE_NESTED, .type_system = &rtnl_address_type_system, .size = sizeof(struct ifaddrmsg) }, +        [RTM_NEWROUTE] = { .type = NETLINK_TYPE_NESTED, .type_system = &rtnl_route_type_system, .size = sizeof(struct rtmsg) }, +        [RTM_DELROUTE] = { .type = NETLINK_TYPE_NESTED, .type_system = &rtnl_route_type_system, .size = sizeof(struct rtmsg) }, +        [RTM_GETROUTE] = { .type = NETLINK_TYPE_NESTED, .type_system = &rtnl_route_type_system, .size = sizeof(struct rtmsg) }, +        [RTM_NEWNEIGH] = { .type = NETLINK_TYPE_NESTED, .type_system = &rtnl_neigh_type_system, .size = sizeof(struct ndmsg) }, +        [RTM_DELNEIGH] = { .type = NETLINK_TYPE_NESTED, .type_system = &rtnl_neigh_type_system, .size = sizeof(struct ndmsg) }, +        [RTM_GETNEIGH] = { .type = NETLINK_TYPE_NESTED, .type_system = &rtnl_neigh_type_system, .size = sizeof(struct ndmsg) }, +}; + +const NLTypeSystem type_system_root = { +        .count = ELEMENTSOF(rtnl_types),          .types = rtnl_types,  }; -int type_system_get_type(const NLTypeSystem *type_system, const NLType **ret, uint16_t type) { -        const NLType *nl_type; +uint16_t type_get_type(const NLType *type) { +        assert(type); +        return type->type; +} +size_t type_get_size(const NLType *type) { +        assert(type); +        return type->size; +} + +void type_get_type_system(const NLType *nl_type, const NLTypeSystem **ret) { +        assert(nl_type);          assert(ret); +        assert(nl_type->type == NETLINK_TYPE_NESTED); +        assert(nl_type->type_system); -        if (!type_system) -                type_system = &rtnl_type_system; +        *ret = nl_type->type_system; +} +void type_get_type_system_union(const NLType *nl_type, const NLTypeSystemUnion **ret) { +        assert(nl_type); +        assert(ret); +        assert(nl_type->type == NETLINK_TYPE_UNION); +        assert(nl_type->type_system_union); + +        *ret = nl_type->type_system_union; +} + +uint16_t type_system_get_count(const NLTypeSystem *type_system) { +        assert(type_system); +        return type_system->count; +} + +int type_system_get_type(const NLTypeSystem *type_system, const NLType **ret, uint16_t type) { +        const NLType *nl_type; + +        assert(ret); +        assert(type_system);          assert(type_system->types); -        if (type > type_system->max) +        if (type >= type_system->count)                  return -EOPNOTSUPP;          nl_type = &type_system->types[type]; -        if (nl_type->type == NLA_UNSPEC) +        if (nl_type->type == NETLINK_TYPE_UNSPEC)                  return -EOPNOTSUPP;          *ret = nl_type; @@ -493,11 +579,7 @@ int type_system_get_type_system(const NLTypeSystem *type_system, const NLTypeSys          if (r < 0)                  return r; -        assert(nl_type->type == NLA_NESTED); -        assert(nl_type->type_system); - -        *ret = nl_type->type_system; - +        type_get_type_system(nl_type, ret);          return 0;  } @@ -511,11 +593,7 @@ int type_system_get_type_system_union(const NLTypeSystem *type_system, const NLT          if (r < 0)                  return r; -        assert(nl_type->type == NLA_UNION); -        assert(nl_type->type_system_union); - -        *ret = nl_type->type_system_union; - +        type_get_type_system_union(nl_type, ret);          return 0;  } @@ -552,7 +630,7 @@ int type_system_union_protocol_get_type_system(const NLTypeSystemUnion *type_sys                  return -EOPNOTSUPP;          type_system = &type_system_union->type_systems[protocol]; -        if (type_system->max == 0) +        if (!type_system->types)                  return -EOPNOTSUPP;          *ret = type_system; diff --git a/src/libsystemd/sd-netlink/netlink-types.h b/src/libsystemd/sd-netlink/netlink-types.h index de1544bf36..a210163241 100644 --- a/src/libsystemd/sd-netlink/netlink-types.h +++ b/src/libsystemd/sd-netlink/netlink-types.h @@ -22,18 +22,17 @@  ***/  enum { -        NLA_UNSPEC, -        NLA_META, -        NLA_U8, -        NLA_U16, -        NLA_U32, -        NLA_U64, -        NLA_STRING, -        NLA_IN_ADDR, -        NLA_ETHER_ADDR, -        NLA_CACHE_INFO, -        NLA_NESTED, -        NLA_UNION, +        NETLINK_TYPE_UNSPEC, +        NETLINK_TYPE_U8,                        /* NLA_U8 */ +        NETLINK_TYPE_U16,                       /* NLA_U16 */ +        NETLINK_TYPE_U32,                       /* NLA_U32 */ +        NETLINK_TYPE_U64,                       /* NLA_U64 */ +        NETLINK_TYPE_STRING,                    /* NLA_STRING */ +        NETLINK_TYPE_IN_ADDR, +        NETLINK_TYPE_ETHER_ADDR, +        NETLINK_TYPE_CACHE_INFO, +        NETLINK_TYPE_NESTED,                    /* NLA_NESTED */ +        NETLINK_TYPE_UNION,  };  typedef enum NLMatchType { @@ -53,18 +52,14 @@ struct NLTypeSystemUnion {          const NLTypeSystem *type_systems;  }; -struct NLTypeSystem { -        uint16_t max; -        const NLType *types; -}; +extern const NLTypeSystem type_system_root; -struct NLType { -        uint16_t type; -        size_t size; -        const NLTypeSystem *type_system; -        const NLTypeSystemUnion *type_system_union; -}; +uint16_t type_get_type(const NLType *type); +size_t type_get_size(const NLType *type); +void type_get_type_system(const NLType *type, const NLTypeSystem **ret); +void type_get_type_system_union(const NLType *type, const NLTypeSystemUnion **ret); +uint16_t type_system_get_count(const NLTypeSystem *type_system);  int type_system_get_type(const NLTypeSystem *type_system, const NLType **ret, uint16_t type);  int type_system_get_type_system(const NLTypeSystem *type_system, const NLTypeSystem **ret, uint16_t type);  int type_system_get_type_system_union(const NLTypeSystem *type_system, const NLTypeSystemUnion **ret, uint16_t type); @@ -95,25 +90,3 @@ typedef enum NLUnionLinkInfoData {  const char *nl_union_link_info_data_to_string(NLUnionLinkInfoData p) _const_;  NLUnionLinkInfoData nl_union_link_info_data_from_string(const char *p) _pure_; - -/* Maximum ARP IP target defined in kernel */ -#define BOND_MAX_ARP_TARGETS    16 - -typedef enum BondArpTargets { -        BOND_ARP_TARGETS_0, -        BOND_ARP_TARGETS_1, -        BOND_ARP_TARGETS_2, -        BOND_ARP_TARGETS_3, -        BOND_ARP_TARGETS_4, -        BOND_ARP_TARGETS_5, -        BOND_ARP_TARGETS_6, -        BOND_ARP_TARGETS_7, -        BOND_ARP_TARGETS_8, -        BOND_ARP_TARGETS_9, -        BOND_ARP_TARGETS_10, -        BOND_ARP_TARGETS_11, -        BOND_ARP_TARGETS_12, -        BOND_ARP_TARGETS_13, -        BOND_ARP_TARGETS_14, -        BOND_ARP_TARGETS_MAX = BOND_MAX_ARP_TARGETS, -} BondArpTargets; | 
