diff options
author | Jonas Gorski <jonas.gorski@bisdn.de> | 2024-02-20 12:50:48 (GMT) |
---|---|---|
committer | Thomas Haller <thaller@redhat.com> | 2024-05-17 14:31:12 (GMT) |
commit | 3e0806317b5116bca7a097a0cb9b7a9cbebc465a (patch) | |
tree | 5132a1dc0a03cfe3b88d4e80a3ad04f9f8e7934f | |
parent | 401c2488624f2eb184e2b6f2e55d07c840854882 (diff) | |
download | libnl-3e0806317b5116bca7a097a0cb9b7a9cbebc465a.zip libnl-3e0806317b5116bca7a097a0cb9b7a9cbebc465a.tar.gz libnl-3e0806317b5116bca7a097a0cb9b7a9cbebc465a.tar.bz2 |
route: expose nexthop id attribute
Routes may reference a nexthop (group) via the new nexthop API by its
ID, so add accessors for setting and getting it.
Referencing a nexthop is mutually exclusive to specifiying nexthops in
the route, so make sure we do not do that when creating netlink
messages (which may exist both, since netlink messages from the kernel
contain both unless 'nexthop_compat_mode' is disabled).
$ ip -6 r
2001:db8:3::/64 nhid 20 metric 1024 pref medium
nexthop via 2001:db8:1::2 dev v0 weight 1
nexthop via 2001:db8:2::2 dev v1 weight 1
Before:
$ nl-route-list
inet6 2001:db8:3::/64 table main type unicast via 2001:db8:1::2 dev v0 via 2001:db8:2::2 dev v1
After:
$ nl-route-list
inet6 2001:db8:3::/64 table main type unicast nhid 20 via 2001:db8:1::2 dev v0 via 2001:db8:2::2 dev v1
Signed-off-by: Jonas Gorski <jonas.gorski@bisdn.de>
https://github.com/thom311/libnl/pull/386
-rw-r--r-- | include/netlink/route/route.h | 2 | ||||
-rw-r--r-- | lib/route/route_obj.c | 35 | ||||
-rw-r--r-- | libnl-route-3.sym | 2 |
3 files changed, 38 insertions, 1 deletions
diff --git a/include/netlink/route/route.h b/include/netlink/route/route.h index 3824762..4377062 100644 --- a/include/netlink/route/route.h +++ b/include/netlink/route/route.h @@ -94,6 +94,8 @@ extern int rtnl_route_get_src_len(struct rtnl_route *); extern void rtnl_route_set_ttl_propagate(struct rtnl_route *route, uint8_t ttl_prop); extern int rtnl_route_get_ttl_propagate(struct rtnl_route *route); +extern void rtnl_route_set_nhid(struct rtnl_route *, uint32_t); +extern uint32_t rtnl_route_get_nhid(struct rtnl_route *); extern void rtnl_route_add_nexthop(struct rtnl_route *, struct rtnl_nexthop *); diff --git a/lib/route/route_obj.c b/lib/route/route_obj.c index 5e77335..6ff110a 100644 --- a/lib/route/route_obj.c +++ b/lib/route/route_obj.c @@ -63,6 +63,7 @@ struct rtnl_route { uint32_t rt_metrics[RTAX_MAX]; uint32_t rt_metrics_mask; uint32_t rt_nr_nh; + uint32_t rt_nhid; struct nl_addr *rt_pref_src; struct nl_list_head rt_nexthops; struct rtnl_rtcacheinfo rt_cacheinfo; @@ -88,6 +89,7 @@ struct rtnl_route { #define ROUTE_ATTR_REALMS 0x010000 #define ROUTE_ATTR_CACHEINFO 0x020000 #define ROUTE_ATTR_TTL_PROPAGATE 0x040000 +#define ROUTE_ATTR_NHID 0x080000 /** @endcond */ static void route_constructor(struct nl_object *c) @@ -192,6 +194,9 @@ static void route_dump_line(struct nl_object *a, struct nl_dump_params *p) if (r->ce_mask & ROUTE_ATTR_TOS && r->rt_tos != 0) nl_dump(p, "tos %#x ", r->rt_tos); + if (r->ce_mask & ROUTE_ATTR_NHID) + nl_dump(p, "nhid %u ", r->rt_nhid); + if (r->ce_mask & ROUTE_ATTR_MULTIPATH) { struct rtnl_nexthop *nh; @@ -282,6 +287,9 @@ static void route_dump_details(struct nl_object *a, struct nl_dump_params *p) r->rt_ttl_propagate ? "enabled" : "disabled"); } + if (r->ce_mask & ROUTE_ATTR_NHID) + nl_dump(p, "nhid %u ", r->rt_nhid); + nl_dump(p, "\n"); if (r->ce_mask & ROUTE_ATTR_MULTIPATH) { @@ -412,6 +420,7 @@ static uint64_t route_compare(struct nl_object *_a, struct nl_object *_b, nl_addr_cmp(a->rt_pref_src, b->rt_pref_src)); diff |= _DIFF(ROUTE_ATTR_TTL_PROPAGATE, a->rt_ttl_propagate != b->rt_ttl_propagate); + diff |= _DIFF(ROUTE_ATTR_NHID, a->rt_nhid != b->rt_nhid); if (flags & LOOSE_COMPARISON) { nl_list_for_each_entry(nh_b, &b->rt_nexthops, rtnh_list) { @@ -617,6 +626,7 @@ static const struct trans_tbl route_attrs[] = { __ADD(ROUTE_ATTR_REALMS, realms), __ADD(ROUTE_ATTR_CACHEINFO, cacheinfo), __ADD(ROUTE_ATTR_TTL_PROPAGATE, ttl_propagate), + __ADD(ROUTE_ATTR_NHID, nhid), }; static char *route_attrs2str(int attrs, char *buf, size_t len) @@ -964,6 +974,21 @@ int rtnl_route_get_ttl_propagate(struct rtnl_route *route) return route->rt_ttl_propagate; } +void rtnl_route_set_nhid(struct rtnl_route *route, uint32_t nhid) +{ + route->rt_nhid = nhid; + + if (nhid > 0) + route->ce_mask |= ROUTE_ATTR_NHID; + else + route->ce_mask &= ~ROUTE_ATTR_NHID; +} + +uint32_t rtnl_route_get_nhid(struct rtnl_route *route) +{ + return route->rt_nhid; +} + /** @} */ /** @@ -1046,6 +1071,7 @@ static struct nla_policy route_policy[RTA_MAX+1] = { [RTA_TTL_PROPAGATE] = { .type = NLA_U8 }, [RTA_ENCAP] = { .type = NLA_NESTED }, [RTA_ENCAP_TYPE] = { .type = NLA_U16 }, + [RTA_NH_ID] = { .type = NLA_U32 }, }; static int parse_multipath(struct rtnl_route *route, struct nlattr *attr) @@ -1337,6 +1363,10 @@ int rtnl_route_parse(struct nlmsghdr *nlh, struct rtnl_route **result) return err; } + if (tb[RTA_NH_ID]) { + rtnl_route_set_nhid(route, nla_get_u32(tb[RTA_NH_ID])); + } + if (old_nh) { rtnl_route_nh_set_flags(old_nh, rtm->rtm_flags & 0xff); if (route->rt_nr_nh == 0) { @@ -1439,7 +1469,10 @@ int rtnl_route_build_msg(struct nl_msg *msg, struct rtnl_route *route) nla_nest_end(msg, metrics); } - if (rtnl_route_get_nnexthops(route) == 1) { + /* Nexthop specification and nexthop id are mutually exclusive */ + if (route->ce_mask & ROUTE_ATTR_NHID) { + NLA_PUT_U32(msg, RTA_NH_ID, route->rt_nhid); + } else if (rtnl_route_get_nnexthops(route) == 1) { struct rtnl_nexthop *nh; nh = rtnl_route_nexthop_n(route, 0); diff --git a/libnl-route-3.sym b/libnl-route-3.sym index aca02e9..9cb21f7 100644 --- a/libnl-route-3.sym +++ b/libnl-route-3.sym @@ -1328,5 +1328,7 @@ global: rtnl_link_bridge_set_port_vlan_map_range; rtnl_link_bridge_set_port_vlan_pvid; rtnl_link_bridge_unset_port_vlan_map_range; + rtnl_route_get_nhid; rtnl_route_nh_identical; + rtnl_route_set_nhid; } libnl_3_9; |