13#include <netlink-private/netlink.h>
14#include <netlink-private/utils.h>
15#include <netlink/netlink.h>
16#include <netlink/attr.h>
17#include <netlink/utils.h>
18#include <netlink/object.h>
19#include <netlink/route/rtnl.h>
20#include <netlink/route/route.h>
21#include <netlink/fib_lookup/request.h>
22#include <netlink/fib_lookup/lookup.h>
25static struct nl_cache_ops fib_lookup_ops;
26static struct nl_object_ops result_obj_ops;
33 unsigned char fl_scope;
34 unsigned char tb_id_in;
37 unsigned char prefixlen;
45static void result_free_data(
struct nl_object *obj)
47 struct flnl_result *res = nl_object_priv(obj);
49 if (res && res->fr_req)
53static int result_clone(
struct nl_object *_dst,
struct nl_object *_src)
55 struct flnl_result *dst = nl_object_priv(_dst);
56 struct flnl_result *src = nl_object_priv(_src);
61 if (!(dst->fr_req = (
struct flnl_request *)
nl_object_clone(OBJ_CAST(src->fr_req))))
68static int result_msg_parser(
struct nl_cache_ops *ops,
struct sockaddr_nl *who,
69 struct nlmsghdr *n,
struct nl_parser_param *pp)
71 struct flnl_result *res;
72 struct fib_result_nl *fr;
76 res = flnl_result_alloc();
80 res->ce_msgtype = n->nlmsg_type;
82 res->fr_req = flnl_request_alloc();
90 err = flnl_request_set_addr(res->fr_req, addr);
95 flnl_request_set_fwmark(res->fr_req, fr->fl_fwmark);
96 flnl_request_set_tos(res->fr_req, fr->fl_tos);
97 flnl_request_set_scope(res->fr_req, fr->fl_scope);
98 flnl_request_set_table(res->fr_req, fr->tb_id_in);
100 res->fr_table_id = fr->tb_id;
101 res->fr_prefixlen = fr->prefixlen;
102 res->fr_nh_sel = fr->nh_sel;
103 res->fr_type = fr->type;
104 res->fr_scope = fr->scope;
105 res->fr_error = fr->err;
107 err = pp->pp_cb((
struct nl_object *) res, pp);
117 flnl_result_put(res);
121static void result_dump_line(
struct nl_object *obj,
struct nl_dump_params *p)
123 struct flnl_result *res = (
struct flnl_result *) obj;
126 nl_dump_line(p,
"table %s prefixlen %u next-hop-selector %u\n",
127 rtnl_route_table2str(res->fr_table_id, buf,
sizeof(buf)),
128 res->fr_prefixlen, res->fr_nh_sel);
129 nl_dump_line(p,
"type %s ",
130 nl_rtntype2str(res->fr_type, buf,
sizeof(buf)));
131 nl_dump(p,
"scope %s error %s (%d)\n",
132 rtnl_scope2str(res->fr_scope, buf,
sizeof(buf)),
133 nl_strerror_l(-res->fr_error), res->fr_error);
136static void result_dump_details(
struct nl_object *obj,
struct nl_dump_params *p)
138 result_dump_line(obj, p);
141static uint64_t result_compare(
struct nl_object *_a,
struct nl_object *_b,
142 uint64_t attrs,
int flags)
152struct flnl_result *flnl_result_alloc(
void)
157void flnl_result_put(
struct flnl_result *res)
206 struct nl_msg **result)
209 struct nl_addr *addr;
211 int tos, scope, table;
212 struct fib_result_nl fr = {0};
214 fwmark = flnl_request_get_fwmark(req);
215 tos = flnl_request_get_tos(req);
216 scope = flnl_request_get_scope(req);
217 table = flnl_request_get_table(req);
219 fr.fl_fwmark = fwmark != UINT_LEAST64_MAX ? fwmark : 0;
220 fr.fl_tos = tos >= 0 ? tos : 0;
221 fr.fl_scope = scope >= 0 ? scope : RT_SCOPE_UNIVERSE;
222 fr.tb_id_in = table >= 0 ? table : RT_TABLE_UNSPEC;
224 addr = flnl_request_get_addr(req);
226 return -NLE_MISSING_ATTR;
234 if (
nlmsg_append(msg, &fr,
sizeof(fr), NLMSG_ALIGNTO) < 0)
257 struct nl_cache *cache)
280int flnl_result_get_table_id(
struct flnl_result *res)
282 return res->fr_table_id;
285int flnl_result_get_prefixlen(
struct flnl_result *res)
287 return res->fr_prefixlen;
290int flnl_result_get_nexthop_sel(
struct flnl_result *res)
292 return res->fr_nh_sel;
295int flnl_result_get_type(
struct flnl_result *res)
300int flnl_result_get_scope(
struct flnl_result *res)
302 return res->fr_scope;
305int flnl_result_get_error(
struct flnl_result *res)
307 return res->fr_error;
312static struct nl_object_ops result_obj_ops = {
313 .oo_name =
"fib_lookup/result",
314 .oo_size =
sizeof(
struct flnl_result),
315 .oo_free_data = result_free_data,
316 .oo_clone = result_clone,
321 .oo_compare = result_compare,
324static struct nl_cache_ops fib_lookup_ops = {
325 .co_name =
"fib_lookup/fib_lookup",
326 .co_hdrsize =
sizeof(
struct fib_result_nl),
328 { 0, NL_ACT_UNSPEC,
"any" },
329 END_OF_MSGTYPES_LIST,
331 .co_protocol = NETLINK_FIB_LOOKUP,
332 .co_msg_parser = result_msg_parser,
333 .co_obj_ops = &result_obj_ops,
336static void __init fib_lookup_init(
void)
341static void __exit fib_lookup_exit(
void)
struct nl_addr * nl_addr_build(int family, const void *buf, size_t size)
Allocate abstract address based on a binary address.
void * nl_addr_get_binary_addr(const struct nl_addr *addr)
Get binary address of abstract address object.
void nl_addr_put(struct nl_addr *addr)
Decrease the reference counter of an abstract address.
int nl_cache_mngt_unregister(struct nl_cache_ops *ops)
Unregister a set of cache operations.
int nl_cache_mngt_register(struct nl_cache_ops *ops)
Register a set of cache operations.
struct nl_cache * nl_cache_alloc(struct nl_cache_ops *ops)
Allocate new cache.
int nl_cache_pickup_checkdup(struct nl_sock *sk, struct nl_cache *cache)
Pickup a netlink dump response and put it into a cache.
@ NL_STOP
Stop parsing altogether and discard remaining messages.
struct nl_cache * flnl_result_alloc_cache(void)
Allocate lookup result cache.
int flnl_lookup_build_request(struct flnl_request *req, int flags, struct nl_msg **result)
Builds a netlink request message to do a lookup.
int flnl_lookup(struct nl_sock *sk, struct flnl_request *req, struct nl_cache *cache)
Perform FIB Lookup.
struct nl_msg * nlmsg_alloc_simple(int nlmsgtype, int flags)
Allocate a new netlink message.
void * nlmsg_data(const struct nlmsghdr *nlh)
Return pointer to message payload.
void nlmsg_free(struct nl_msg *msg)
Release a reference from an netlink message.
int nlmsg_append(struct nl_msg *n, void *data, size_t len, int pad)
Append data to tail of a netlink message.
struct nl_object * nl_object_clone(struct nl_object *obj)
Allocate a new object and copy all data from an existing object.
void nl_object_put(struct nl_object *obj)
Release a reference from an object.
struct nl_object * nl_object_alloc(struct nl_object_ops *ops)
Allocate a new object of kind specified by the operations handle.
int nl_send_auto_complete(struct nl_sock *sk, struct nl_msg *msg)
void nl_dump(struct nl_dump_params *params, const char *fmt,...)
Dump a formatted character string.
@ NL_DUMP_LINE
Dump object briefly on one line.
@ NL_DUMP_DETAILS
Dump all attributes but no statistics.