/* * Copyright (c) 2004-2021 Apple Inc. All rights reserved. * * @APPLE_OSREFERENCE_LICENSE_HEADER_START@ * * This file contains Original Code and/or Modifications of Original Code * as defined in and that are subject to the Apple Public Source License * Version 2.0 (the 'License'). You may not use this file except in * compliance with the License. The rights granted to you under the License * may not be used to create, or enable the creation or redistribution of, * unlawful or unlicensed copies of an Apple operating system, or to * circumvent, violate, or enable the circumvention or violation of, any * terms of an Apple operating system software license agreement. * * Please obtain a copy of the License at * http://www.opensource.apple.com/apsl/ and read it before using this file. * * The Original Code and all software distributed under the License are * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES, * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT. * Please see the License for the specific language governing rights and * limitations under the License. * * @APPLE_OSREFERENCE_LICENSE_HEADER_END@ */ #include "kpi_protocol.h" #include #include #include #include #include #include #include #include #include void proto_input_run(void); typedef int (*attach_t)(struct ifnet *ifp, uint32_t protocol_family); typedef int (*detach_t)(struct ifnet *ifp, uint32_t protocol_family); struct proto_input_entry { struct proto_input_entry *next; int detach; struct domain *domain; int hash; int chain; protocol_family_t protocol; proto_input_handler input; proto_input_detached_handler detached; mbuf_t inject_first; mbuf_t inject_last; struct proto_input_entry *input_next; mbuf_t input_first; mbuf_t input_last; }; struct proto_family_str { TAILQ_ENTRY(proto_family_str) proto_fam_next; protocol_family_t proto_family; ifnet_family_t if_family; proto_plumb_handler attach_proto; proto_unplumb_handler detach_proto; }; static LCK_GRP_DECLARE(proto_family_grp, "protocol kpi"); static LCK_MTX_DECLARE(proto_family_mutex, &proto_family_grp); static struct proto_input_entry *proto_hash[PROTO_HASH_SLOTS]; static int proto_total_waiting = 0; static struct proto_input_entry *proto_input_add_list = NULL; static TAILQ_HEAD(, proto_family_str) proto_family_head = TAILQ_HEAD_INITIALIZER(proto_family_head); __private_extern__ errno_t proto_register_input(protocol_family_t protocol, proto_input_handler input, proto_input_detached_handler detached, int chains) { struct proto_input_entry *entry; struct dlil_threading_info *inp = dlil_main_input_thread; struct domain *dp; domain_guard_t guard; entry = kalloc_type(struct proto_input_entry, Z_WAITOK | Z_ZERO); if (entry == NULL) { return ENOMEM; } entry->protocol = protocol; entry->input = input; entry->detached = detached; entry->hash = proto_hash_value(protocol); entry->chain = chains; guard = domain_guard_deploy(); TAILQ_FOREACH(dp, &domains, dom_entry) { if (dp->dom_family == (int)protocol) { break; } } domain_guard_release(guard); if (dp == NULL) { return EINVAL; } entry->domain = dp; lck_mtx_lock(&inp->dlth_lock); entry->next = proto_input_add_list; proto_input_add_list = entry; inp->dlth_flags |= DLIL_PROTO_REGISTER; if ((inp->dlth_flags & DLIL_INPUT_RUNNING) == 0) { wakeup((caddr_t)&inp->dlth_flags); } lck_mtx_unlock(&inp->dlth_lock); return 0; } __private_extern__ void proto_unregister_input(protocol_family_t protocol) { struct proto_input_entry *entry = NULL; for (entry = proto_hash[proto_hash_value(protocol)]; entry != NULL; entry = entry->next) { if (entry->protocol == protocol) { break; } } if (entry != NULL) { entry->detach = 1; } } static void proto_delayed_attach(struct proto_input_entry *entry) { struct proto_input_entry *next_entry; for (next_entry = entry->next; entry != NULL; entry = next_entry) { struct proto_input_entry *exist; int hash_slot; hash_slot = proto_hash_value(entry->protocol); next_entry = entry->next; for (exist = proto_hash[hash_slot]; exist != NULL; exist = exist->next) { if (exist->protocol == entry->protocol) { break; } } /* If the entry already exists, call detached and dispose */ if (exist != NULL) { if (entry->detached) { entry->detached(entry->protocol); } kfree_type(struct proto_input_entry, entry); } else { entry->next = proto_hash[hash_slot]; proto_hash[hash_slot] = entry; } } } __private_extern__ void proto_input_run(void) { struct proto_input_entry *entry; struct dlil_threading_info *inp = dlil_main_input_thread; mbuf_t packet_list; int i, locked = 0; LCK_MTX_ASSERT(&inp->dlth_lock, LCK_MTX_ASSERT_NOTOWNED); if (inp->dlth_flags & DLIL_PROTO_REGISTER) { lck_mtx_lock_spin(&inp->dlth_lock); entry = proto_input_add_list; proto_input_add_list = NULL; inp->dlth_flags &= ~DLIL_PROTO_REGISTER; lck_mtx_unlock(&inp->dlth_lock); proto_delayed_attach(entry); } /* * Move everything from the lock protected list to the thread * specific list. */ for (i = 0; proto_total_waiting != 0 && i < PROTO_HASH_SLOTS; i++) { for (entry = proto_hash[i]; entry != NULL && proto_total_waiting; entry = entry->next) { if (entry->inject_first != NULL) { lck_mtx_lock_spin(&inp->dlth_lock); inp->dlth_flags &= ~DLIL_PROTO_WAITING; packet_list = entry->inject_first; entry->inject_first = NULL; entry->inject_last = NULL; proto_total_waiting--; lck_mtx_unlock(&inp->dlth_lock); if (entry->domain != NULL && !(entry->domain-> dom_flags & DOM_REENTRANT)) { lck_mtx_lock(entry->domain->dom_mtx); locked = 1; } if (entry->chain) { entry->input(entry->protocol, packet_list); } else { mbuf_t packet; for (packet = packet_list; packet != NULL; packet = packet_list) { packet_list = mbuf_nextpkt(packet); mbuf_setnextpkt(packet, NULL); entry->input(entry->protocol, packet); } } if (locked) { locked = 0; lck_mtx_unlock(entry->domain->dom_mtx); } } } } } errno_t proto_input(protocol_family_t protocol, mbuf_t packet_list) { struct proto_input_entry *entry; errno_t locked = 0, result = 0; for (entry = proto_hash[proto_hash_value(protocol)]; entry != NULL; entry = entry->next) { if (entry->protocol == protocol) { break; } } if (entry == NULL) { return -1; } if (entry->domain && !(entry->domain->dom_flags & DOM_REENTRANT)) { lck_mtx_lock(entry->domain->dom_mtx); locked = 1; } if (entry->chain) { entry->input(entry->protocol, packet_list); } else { mbuf_t packet; for (packet = packet_list; packet != NULL; packet = packet_list) { packet_list = mbuf_nextpkt(packet); mbuf_setnextpkt(packet, NULL); entry->input(entry->protocol, packet); } } if (locked) { lck_mtx_unlock(entry->domain->dom_mtx); } return result; } errno_t proto_inject(protocol_family_t protocol, mbuf_t packet_list) { struct proto_input_entry *entry; mbuf_t last_packet; int hash_slot = proto_hash_value(protocol); struct dlil_threading_info *inp = dlil_main_input_thread; for (last_packet = packet_list; mbuf_nextpkt(last_packet) != NULL; last_packet = mbuf_nextpkt(last_packet)) { /* find the last packet */; } for (entry = proto_hash[hash_slot]; entry != NULL; entry = entry->next) { if (entry->protocol == protocol) { break; } } if (entry != NULL) { lck_mtx_lock(&inp->dlth_lock); if (entry->inject_first == NULL) { proto_total_waiting++; inp->dlth_flags |= DLIL_PROTO_WAITING; entry->inject_first = packet_list; } else { mbuf_setnextpkt(entry->inject_last, packet_list); } entry->inject_last = last_packet; if ((inp->dlth_flags & DLIL_INPUT_RUNNING) == 0) { wakeup((caddr_t)&inp->dlth_flags); } lck_mtx_unlock(&inp->dlth_lock); } else { return ENOENT; } return 0; } static struct proto_family_str * proto_plumber_find(protocol_family_t proto_family, ifnet_family_t if_family) { struct proto_family_str *mod = NULL; TAILQ_FOREACH(mod, &proto_family_head, proto_fam_next) { if ((mod->proto_family == (proto_family & 0xffff)) && (mod->if_family == (if_family & 0xffff))) { break; } } return mod; } errno_t proto_register_plumber(protocol_family_t protocol_family, ifnet_family_t interface_family, proto_plumb_handler attach, proto_unplumb_handler detach) { struct proto_family_str *proto_family; if (attach == NULL) { return EINVAL; } lck_mtx_lock(&proto_family_mutex); TAILQ_FOREACH(proto_family, &proto_family_head, proto_fam_next) { if (proto_family->proto_family == protocol_family && proto_family->if_family == interface_family) { lck_mtx_unlock(&proto_family_mutex); return EEXIST; } } proto_family = kalloc_type(struct proto_family_str, Z_WAITOK | Z_ZERO | Z_NOFAIL); proto_family->proto_family = protocol_family; proto_family->if_family = interface_family & 0xffff; proto_family->attach_proto = attach; proto_family->detach_proto = detach; TAILQ_INSERT_TAIL(&proto_family_head, proto_family, proto_fam_next); lck_mtx_unlock(&proto_family_mutex); return 0; } void proto_unregister_plumber(protocol_family_t protocol_family, ifnet_family_t interface_family) { struct proto_family_str *proto_family; lck_mtx_lock(&proto_family_mutex); proto_family = proto_plumber_find(protocol_family, interface_family); if (proto_family == NULL) { lck_mtx_unlock(&proto_family_mutex); return; } TAILQ_REMOVE(&proto_family_head, proto_family, proto_fam_next); kfree_type(struct proto_family_str, proto_family); lck_mtx_unlock(&proto_family_mutex); } __private_extern__ errno_t proto_plumb(protocol_family_t protocol_family, ifnet_t ifp) { struct proto_family_str *proto_family; int ret = 0; lck_mtx_lock(&proto_family_mutex); proto_family = proto_plumber_find(protocol_family, ifp->if_family); if (proto_family == NULL) { lck_mtx_unlock(&proto_family_mutex); return ENXIO; } ret = proto_family->attach_proto(ifp, protocol_family); lck_mtx_unlock(&proto_family_mutex); return ret; } __private_extern__ errno_t proto_unplumb(protocol_family_t protocol_family, ifnet_t ifp) { struct proto_family_str *proto_family; int ret = 0; lck_mtx_lock(&proto_family_mutex); proto_family = proto_plumber_find(protocol_family, ifp->if_family); if (proto_family != NULL && proto_family->detach_proto) { proto_family->detach_proto(ifp, protocol_family); } else { ret = ifnet_detach_protocol(ifp, protocol_family); } lck_mtx_unlock(&proto_family_mutex); return ret; }