1a6663252SAlexander V. Chernikov /*- 2a6663252SAlexander V. Chernikov * SPDX-License-Identifier: BSD-2-Clause-FreeBSD 3a6663252SAlexander V. Chernikov * 4a6663252SAlexander V. Chernikov * Copyright (c) 2020 Alexander V. Chernikov 5a6663252SAlexander V. Chernikov * 6a6663252SAlexander V. Chernikov * Redistribution and use in source and binary forms, with or without 7a6663252SAlexander V. Chernikov * modification, are permitted provided that the following conditions 8a6663252SAlexander V. Chernikov * are met: 9a6663252SAlexander V. Chernikov * 1. Redistributions of source code must retain the above copyright 10a6663252SAlexander V. Chernikov * notice, this list of conditions and the following disclaimer. 11a6663252SAlexander V. Chernikov * 2. Redistributions in binary form must reproduce the above copyright 12a6663252SAlexander V. Chernikov * notice, this list of conditions and the following disclaimer in the 13a6663252SAlexander V. Chernikov * documentation and/or other materials provided with the distribution. 14a6663252SAlexander V. Chernikov * 15a6663252SAlexander V. Chernikov * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 16a6663252SAlexander V. Chernikov * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 17a6663252SAlexander V. Chernikov * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 18a6663252SAlexander V. Chernikov * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 19a6663252SAlexander V. Chernikov * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 20a6663252SAlexander V. Chernikov * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 21a6663252SAlexander V. Chernikov * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 22a6663252SAlexander V. Chernikov * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 23a6663252SAlexander V. Chernikov * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 24a6663252SAlexander V. Chernikov * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 25a6663252SAlexander V. Chernikov * SUCH DAMAGE. 26a6663252SAlexander V. Chernikov */ 27a6663252SAlexander V. Chernikov 28a6663252SAlexander V. Chernikov #include <sys/cdefs.h> 29a6663252SAlexander V. Chernikov __FBSDID("$FreeBSD$"); 30a6663252SAlexander V. Chernikov #include "opt_inet.h" 31a6663252SAlexander V. Chernikov #include "opt_inet6.h" 32a6663252SAlexander V. Chernikov #include "opt_route.h" 33a6663252SAlexander V. Chernikov 34a6663252SAlexander V. Chernikov #include <sys/param.h> 35a6663252SAlexander V. Chernikov #include <sys/jail.h> 36a6663252SAlexander V. Chernikov #include <sys/systm.h> 37a6663252SAlexander V. Chernikov #include <sys/malloc.h> 38a6663252SAlexander V. Chernikov #include <sys/mbuf.h> 39a6663252SAlexander V. Chernikov #include <sys/socket.h> 40a6663252SAlexander V. Chernikov #include <sys/sysctl.h> 41a6663252SAlexander V. Chernikov #include <sys/syslog.h> 42a6663252SAlexander V. Chernikov #include <sys/sysproto.h> 43a6663252SAlexander V. Chernikov #include <sys/proc.h> 44a6663252SAlexander V. Chernikov #include <sys/domain.h> 45a6663252SAlexander V. Chernikov #include <sys/kernel.h> 46a6663252SAlexander V. Chernikov #include <sys/lock.h> 47a6663252SAlexander V. Chernikov #include <sys/rmlock.h> 48a6663252SAlexander V. Chernikov 49a6663252SAlexander V. Chernikov #include <net/if.h> 50a6663252SAlexander V. Chernikov #include <net/if_var.h> 51a6663252SAlexander V. Chernikov #include <net/if_dl.h> 52a6663252SAlexander V. Chernikov #include <net/route.h> 53e7d8af4fSAlexander V. Chernikov #include <net/route/route_var.h> 54a6663252SAlexander V. Chernikov #include <net/route/nhop_utils.h> 55a6663252SAlexander V. Chernikov #include <net/route/nhop.h> 56a6663252SAlexander V. Chernikov #include <net/route/nhop_var.h> 57a6663252SAlexander V. Chernikov #include <net/route/shared.h> 58*682b902dSAlexander V. Chernikov #ifdef INET 59*682b902dSAlexander V. Chernikov #include <netinet/in_fib.h> 60*682b902dSAlexander V. Chernikov #endif 61*682b902dSAlexander V. Chernikov #ifdef INET6 62*682b902dSAlexander V. Chernikov #include <netinet6/in6_fib.h> 63*682b902dSAlexander V. Chernikov #endif 64a6663252SAlexander V. Chernikov #include <net/vnet.h> 65a6663252SAlexander V. Chernikov 66a6663252SAlexander V. Chernikov /* 67a6663252SAlexander V. Chernikov * RIB helper functions. 68a6663252SAlexander V. Chernikov */ 69a6663252SAlexander V. Chernikov 70a6663252SAlexander V. Chernikov /* 71a6663252SAlexander V. Chernikov * Calls @wa_f with @arg for each entry in the table specified by 72a6663252SAlexander V. Chernikov * @af and @fibnum. 73a6663252SAlexander V. Chernikov * 74a6663252SAlexander V. Chernikov * Table is traversed under read lock. 75a6663252SAlexander V. Chernikov */ 76a6663252SAlexander V. Chernikov void 77a6663252SAlexander V. Chernikov rib_walk(int af, u_int fibnum, rt_walktree_f_t *wa_f, void *arg) 78a6663252SAlexander V. Chernikov { 79a6663252SAlexander V. Chernikov RIB_RLOCK_TRACKER; 80a6663252SAlexander V. Chernikov struct rib_head *rnh; 81a6663252SAlexander V. Chernikov 82a6663252SAlexander V. Chernikov if ((rnh = rt_tables_get_rnh(fibnum, af)) == NULL) 83a6663252SAlexander V. Chernikov return; 84a6663252SAlexander V. Chernikov 85a6663252SAlexander V. Chernikov RIB_RLOCK(rnh); 86a6663252SAlexander V. Chernikov rnh->rnh_walktree(&rnh->head, (walktree_f_t *)wa_f, arg); 87a6663252SAlexander V. Chernikov RIB_RUNLOCK(rnh); 88a6663252SAlexander V. Chernikov } 89a6663252SAlexander V. Chernikov 90*682b902dSAlexander V. Chernikov /* 91*682b902dSAlexander V. Chernikov * Wrapper for the control plane functions for performing af-agnostic 92*682b902dSAlexander V. Chernikov * lookups. 93*682b902dSAlexander V. Chernikov * @fibnum: fib to perform the lookup. 94*682b902dSAlexander V. Chernikov * @dst: sockaddr with family and addr filled in. IPv6 addresses needs to be in 95*682b902dSAlexander V. Chernikov * deembedded from. 96*682b902dSAlexander V. Chernikov * @flags: fib(9) flags. 97*682b902dSAlexander V. Chernikov * @flowid: flow id for path selection in multipath use case. 98*682b902dSAlexander V. Chernikov * 99*682b902dSAlexander V. Chernikov * Returns nhop_object or NULL. 100*682b902dSAlexander V. Chernikov * 101*682b902dSAlexander V. Chernikov * Requires NET_EPOCH. 102*682b902dSAlexander V. Chernikov * 103*682b902dSAlexander V. Chernikov */ 104*682b902dSAlexander V. Chernikov struct nhop_object * 105*682b902dSAlexander V. Chernikov rib_lookup(uint32_t fibnum, const struct sockaddr *dst, uint32_t flags, 106*682b902dSAlexander V. Chernikov uint32_t flowid) 107*682b902dSAlexander V. Chernikov { 108*682b902dSAlexander V. Chernikov struct nhop_object *nh; 109*682b902dSAlexander V. Chernikov 110*682b902dSAlexander V. Chernikov nh = NULL; 111*682b902dSAlexander V. Chernikov 112*682b902dSAlexander V. Chernikov switch (dst->sa_family) { 113*682b902dSAlexander V. Chernikov #ifdef INET 114*682b902dSAlexander V. Chernikov case AF_INET: 115*682b902dSAlexander V. Chernikov { 116*682b902dSAlexander V. Chernikov const struct sockaddr_in *a = (const struct sockaddr_in *)dst; 117*682b902dSAlexander V. Chernikov nh = fib4_lookup(fibnum, a->sin_addr, 0, flags, flowid); 118*682b902dSAlexander V. Chernikov break; 119*682b902dSAlexander V. Chernikov } 120*682b902dSAlexander V. Chernikov #endif 121*682b902dSAlexander V. Chernikov #ifdef INET6 122*682b902dSAlexander V. Chernikov case AF_INET6: 123*682b902dSAlexander V. Chernikov { 124*682b902dSAlexander V. Chernikov const struct sockaddr_in6 *a = (const struct sockaddr_in6*)dst; 125*682b902dSAlexander V. Chernikov nh = fib6_lookup(fibnum, &a->sin6_addr, a->sin6_scope_id, 126*682b902dSAlexander V. Chernikov flags, flowid); 127*682b902dSAlexander V. Chernikov break; 128*682b902dSAlexander V. Chernikov } 129*682b902dSAlexander V. Chernikov #endif 130*682b902dSAlexander V. Chernikov } 131*682b902dSAlexander V. Chernikov 132*682b902dSAlexander V. Chernikov return (nh); 133*682b902dSAlexander V. Chernikov } 134*682b902dSAlexander V. Chernikov 135