1f5baf8bbSAlexander V. Chernikov /*- 2f5baf8bbSAlexander V. Chernikov * Copyright (c) 2020 3f5baf8bbSAlexander V. Chernikov * Alexander V. Chernikov <melifaro@FreeBSD.org> 4f5baf8bbSAlexander V. Chernikov * 5f5baf8bbSAlexander V. Chernikov * Redistribution and use in source and binary forms, with or without 6f5baf8bbSAlexander V. Chernikov * modification, are permitted provided that the following conditions 7f5baf8bbSAlexander V. Chernikov * are met: 8f5baf8bbSAlexander V. Chernikov * 1. Redistributions of source code must retain the above copyright 9f5baf8bbSAlexander V. Chernikov * notice, this list of conditions and the following disclaimer. 10f5baf8bbSAlexander V. Chernikov * 2. Redistributions in binary form must reproduce the above copyright 11f5baf8bbSAlexander V. Chernikov * notice, this list of conditions and the following disclaimer in the 12f5baf8bbSAlexander V. Chernikov * documentation and/or other materials provided with the distribution. 13f5baf8bbSAlexander V. Chernikov * 3. Neither the name of the University nor the names of its contributors 14f5baf8bbSAlexander V. Chernikov * may be used to endorse or promote products derived from this software 15f5baf8bbSAlexander V. Chernikov * without specific prior written permission. 16f5baf8bbSAlexander V. Chernikov * 17f5baf8bbSAlexander V. Chernikov * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 18f5baf8bbSAlexander V. Chernikov * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 19f5baf8bbSAlexander V. Chernikov * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 20f5baf8bbSAlexander V. Chernikov * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 21f5baf8bbSAlexander V. Chernikov * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 22f5baf8bbSAlexander V. Chernikov * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 23f5baf8bbSAlexander V. Chernikov * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 24f5baf8bbSAlexander V. Chernikov * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 25f5baf8bbSAlexander V. Chernikov * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 26f5baf8bbSAlexander V. Chernikov * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 27f5baf8bbSAlexander V. Chernikov * SUCH DAMAGE. 28f5baf8bbSAlexander V. Chernikov */ 29f5baf8bbSAlexander V. Chernikov 30f5baf8bbSAlexander V. Chernikov 31f5baf8bbSAlexander V. Chernikov struct fib_data; 32f5baf8bbSAlexander V. Chernikov struct fib_dp; 33f5baf8bbSAlexander V. Chernikov enum flm_op_result { 34f5baf8bbSAlexander V. Chernikov FLM_SUCCESS, /* No errors, operation successful */ 35f5baf8bbSAlexander V. Chernikov FLM_REBUILD, /* Operation cannot be completed, schedule algorithm rebuild */ 36f5baf8bbSAlexander V. Chernikov FLM_ERROR, /* Operation failed, this algo cannot be used */ 376b8ef0d4SAlexander V. Chernikov FLM_BATCH, /* Operation cannot be completed, algorithm asks to batch changes */ 38f5baf8bbSAlexander V. Chernikov }; 39f5baf8bbSAlexander V. Chernikov 40f5baf8bbSAlexander V. Chernikov struct rib_rtable_info { 41f5baf8bbSAlexander V. Chernikov uint32_t num_prefixes; 42f5baf8bbSAlexander V. Chernikov uint32_t num_nhops; 43f5baf8bbSAlexander V. Chernikov uint32_t num_nhgrp; 44f5baf8bbSAlexander V. Chernikov }; 45f5baf8bbSAlexander V. Chernikov 46f5baf8bbSAlexander V. Chernikov struct flm_lookup_key { 47f5baf8bbSAlexander V. Chernikov union { 48f5baf8bbSAlexander V. Chernikov const struct in6_addr *addr6; 49f5baf8bbSAlexander V. Chernikov struct in_addr addr4; 50f5baf8bbSAlexander V. Chernikov }; 51f5baf8bbSAlexander V. Chernikov }; 52f5baf8bbSAlexander V. Chernikov 536b8ef0d4SAlexander V. Chernikov struct fib_change_entry { 546b8ef0d4SAlexander V. Chernikov union { 556b8ef0d4SAlexander V. Chernikov struct in_addr addr4; 566b8ef0d4SAlexander V. Chernikov struct in6_addr addr6; 576b8ef0d4SAlexander V. Chernikov }; 586b8ef0d4SAlexander V. Chernikov uint32_t scopeid; 596b8ef0d4SAlexander V. Chernikov uint8_t plen; 606b8ef0d4SAlexander V. Chernikov struct nhop_object *nh_old; 616b8ef0d4SAlexander V. Chernikov struct nhop_object *nh_new; 626b8ef0d4SAlexander V. Chernikov }; 636b8ef0d4SAlexander V. Chernikov 646b8ef0d4SAlexander V. Chernikov struct fib_change_queue { 656b8ef0d4SAlexander V. Chernikov uint32_t count; 666b8ef0d4SAlexander V. Chernikov uint32_t size; 676b8ef0d4SAlexander V. Chernikov struct fib_change_entry *entries; 686b8ef0d4SAlexander V. Chernikov }; 696b8ef0d4SAlexander V. Chernikov 706b8ef0d4SAlexander V. Chernikov 71f5baf8bbSAlexander V. Chernikov typedef struct nhop_object *flm_lookup_t(void *algo_data, 72f5baf8bbSAlexander V. Chernikov const struct flm_lookup_key key, uint32_t scopeid); 73f5baf8bbSAlexander V. Chernikov typedef enum flm_op_result flm_init_t (uint32_t fibnum, struct fib_data *fd, 74f5baf8bbSAlexander V. Chernikov void *_old_data, void **new_data); 75f5baf8bbSAlexander V. Chernikov typedef void flm_destroy_t(void *data); 76f5baf8bbSAlexander V. Chernikov typedef enum flm_op_result flm_dump_t(struct rtentry *rt, void *data); 77f5baf8bbSAlexander V. Chernikov typedef enum flm_op_result flm_dump_end_t(void *data, struct fib_dp *dp); 78f5baf8bbSAlexander V. Chernikov typedef enum flm_op_result flm_change_t(struct rib_head *rnh, 79f5baf8bbSAlexander V. Chernikov struct rib_cmd_info *rc, void *data); 806b8ef0d4SAlexander V. Chernikov typedef enum flm_op_result flm_change_batch_t(struct rib_head *rnh, 816b8ef0d4SAlexander V. Chernikov struct fib_change_queue *q, void *data); 82f5baf8bbSAlexander V. Chernikov typedef uint8_t flm_get_pref_t(const struct rib_rtable_info *rinfo); 83f5baf8bbSAlexander V. Chernikov 84f5baf8bbSAlexander V. Chernikov struct fib_lookup_module { 85f5baf8bbSAlexander V. Chernikov char *flm_name; /* algo name */ 86f5baf8bbSAlexander V. Chernikov int flm_family; /* address family this module supports */ 87f5baf8bbSAlexander V. Chernikov int flm_refcount; /* # of references */ 88f5baf8bbSAlexander V. Chernikov uint32_t flm_flags; /* flags */ 89f5baf8bbSAlexander V. Chernikov uint8_t flm_index; /* internal algo index */ 90f5baf8bbSAlexander V. Chernikov flm_init_t *flm_init_cb; /* instance init */ 91f5baf8bbSAlexander V. Chernikov flm_destroy_t *flm_destroy_cb; /* destroy instance */ 92f5baf8bbSAlexander V. Chernikov flm_change_t *flm_change_rib_item_cb;/* routing table change hook */ 93f5baf8bbSAlexander V. Chernikov flm_dump_t *flm_dump_rib_item_cb; /* routing table dump cb */ 94f5baf8bbSAlexander V. Chernikov flm_dump_end_t *flm_dump_end_cb; /* end of dump */ 95f5baf8bbSAlexander V. Chernikov flm_lookup_t *flm_lookup; /* lookup function */ 96f5baf8bbSAlexander V. Chernikov flm_get_pref_t *flm_get_pref; /* get algo preference */ 976b8ef0d4SAlexander V. Chernikov flm_change_batch_t *flm_change_rib_items_cb;/* routing table change hook */ 986b8ef0d4SAlexander V. Chernikov void *spare[8]; /* Spare callbacks */ 99f5baf8bbSAlexander V. Chernikov TAILQ_ENTRY(fib_lookup_module) entries; 100f5baf8bbSAlexander V. Chernikov }; 101f5baf8bbSAlexander V. Chernikov 102f5baf8bbSAlexander V. Chernikov /* Datapath lookup data */ 103f5baf8bbSAlexander V. Chernikov struct fib_dp { 104f5baf8bbSAlexander V. Chernikov flm_lookup_t *f; 105f5baf8bbSAlexander V. Chernikov void *arg; 106f5baf8bbSAlexander V. Chernikov }; 107f5baf8bbSAlexander V. Chernikov 108f5baf8bbSAlexander V. Chernikov VNET_DECLARE(struct fib_dp *, inet_dp); 109f5baf8bbSAlexander V. Chernikov #define V_inet_dp VNET(inet_dp) 110f5baf8bbSAlexander V. Chernikov VNET_DECLARE(struct fib_dp *, inet6_dp); 111f5baf8bbSAlexander V. Chernikov #define V_inet6_dp VNET(inet6_dp) 112f5baf8bbSAlexander V. Chernikov 113f5baf8bbSAlexander V. Chernikov #define FIB_PRINTF(_l, _fd, _fmt, ...) fib_printf(_l, _fd, __func__, _fmt, ##__VA_ARGS__) 114f5baf8bbSAlexander V. Chernikov 115f5baf8bbSAlexander V. Chernikov void fib_printf(int level, struct fib_data *fd, const char *func, char *fmt, ...); 116f5baf8bbSAlexander V. Chernikov int fib_module_init(struct fib_lookup_module *flm, uint32_t fibnum, 117f5baf8bbSAlexander V. Chernikov int family); 118f5baf8bbSAlexander V. Chernikov int fib_module_clone(const struct fib_lookup_module *flm_orig, 119f5baf8bbSAlexander V. Chernikov struct fib_lookup_module *flm, bool waitok); 120f5baf8bbSAlexander V. Chernikov int fib_module_dumptree(struct fib_lookup_module *flm, 121f5baf8bbSAlexander V. Chernikov enum rib_subscription_type subscription_type); 122f5baf8bbSAlexander V. Chernikov int fib_module_register(struct fib_lookup_module *flm); 123f5baf8bbSAlexander V. Chernikov int fib_module_unregister(struct fib_lookup_module *flm); 124f5baf8bbSAlexander V. Chernikov 125f5baf8bbSAlexander V. Chernikov uint32_t fib_get_nhop_idx(struct fib_data *fd, struct nhop_object *nh); 126f5baf8bbSAlexander V. Chernikov struct nhop_object **fib_get_nhop_array(struct fib_data *fd); 127f5baf8bbSAlexander V. Chernikov void fib_get_rtable_info(struct rib_head *rh, struct rib_rtable_info *rinfo); 128f5baf8bbSAlexander V. Chernikov struct rib_head *fib_get_rh(struct fib_data *fd); 129*e2f79d9eSAlexander V. Chernikov bool fib_set_datapath_ptr(struct fib_data *fd, struct fib_dp *dp); 130*e2f79d9eSAlexander V. Chernikov void fib_set_algo_ptr(struct fib_data *fd, void *algo_data); 131*e2f79d9eSAlexander V. Chernikov void fib_epoch_call(epoch_callback_t callback, epoch_context_t ctx); 132f5baf8bbSAlexander V. Chernikov 133