]> git.saurik.com Git - apple/xnu.git/blobdiff - bsd/net/kpi_interface.c
xnu-1228.12.14.tar.gz
[apple/xnu.git] / bsd / net / kpi_interface.c
index f092a858367aed093dc0bfc0aaf33b460ee7178c..d9dfca3f3eca4e48e740005575d3252a06dab467 100644 (file)
@@ -1,37 +1,36 @@
 /*
- * Copyright (c) 2004 Apple Computer, Inc. All rights reserved.
+ * Copyright (c) 2004-2007 Apple Inc. All rights reserved.
  *
- * @APPLE_LICENSE_OSREFERENCE_HEADER_START@
+ * @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 
+ * 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_LICENSE_OSREFERENCE_HEADER_END@
+ * 
+ * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
  */
 
 #include "kpi_interface.h"
 
 #include <sys/queue.h>
 #include <sys/param.h> /* for definition of NULL */
+#include <kern/debug.h> /* for panic */
 #include <sys/errno.h>
 #include <sys/socket.h>
 #include <sys/kern_event.h>
@@ -45,6 +44,7 @@
 #include <net/if_dl.h>
 #include <net/if_arp.h>
 #include <libkern/libkern.h>
+#include <libkern/OSAtomic.h>
 #include <kern/locks.h>
 
 #if IF_LASTCHANGEUPTIME
 #define TOUCHLASTCHANGE(__if_lastchange) microtime(__if_lastchange)
 #endif
 
-extern lck_spin_t *dlil_input_lock;
+extern struct dlil_threading_info *dlil_lo_thread_ptr;
+extern int dlil_multithreaded_input;
+
+static errno_t
+ifnet_list_get_common(ifnet_family_t, boolean_t, ifnet_t **, u_int32_t *);
 
 /*
        Temporary work around until we have real reference counting
@@ -80,6 +84,17 @@ ifnet_kpi_free(
        dlil_if_release(ifp);
 }
 
+static __inline__ void*
+_cast_non_const(const void * ptr) {
+       union {
+               const void*             cval;
+               void*                   val;
+       } ret;
+       
+       ret.cval = ptr;
+       return (ret.val);
+}
+
 errno_t
 ifnet_allocate(
        const struct ifnet_init_params *init,
@@ -100,14 +115,19 @@ ifnet_allocate(
        
        error = dlil_if_acquire(init->family, init->uniqueid, init->uniqueid_len, &ifp);
        if (error == 0)
-       {               
-               strncpy(ifp->if_name, init->name, IFNAMSIZ);
+       {
+               /*
+                * Cast ifp->if_name as non const. dlil_if_acquire sets it up
+                * to point to storage of at least IFNAMSIZ bytes. It is safe
+                * to write to this.
+                */
+               strncpy(_cast_non_const(ifp->if_name), init->name, IFNAMSIZ);
                ifp->if_type = init->type;
                ifp->if_family = init->family;
                ifp->if_unit = init->unit;
                ifp->if_output = init->output;
                ifp->if_demux = init->demux;
-               ifp->if_add_proto_u.kpi = init->add_proto;
+               ifp->if_add_proto = init->add_proto;
                ifp->if_del_proto = init->del_proto;
                ifp->if_check_multi = init->check_multi;
                ifp->if_framer = init->framer;
@@ -163,49 +183,30 @@ ifnet_allocate(
 
 errno_t
 ifnet_reference(
-       ifnet_t interface)
+       ifnet_t ifp)
 {
-       if (interface == NULL) return EINVAL;
-       ifp_reference(interface);
+       int     oldval;
+       
+       if (ifp == NULL) return EINVAL;
+       
+       oldval = OSIncrementAtomic((SInt32 *)&ifp->if_refcnt);
+       
        return 0;
 }
 
 errno_t
 ifnet_release(
-       ifnet_t interface)
-{
-       if (interface == NULL) return EINVAL;
-       ifp_release(interface);
-       return 0;
-}
-
-errno_t
-ifnet_attach(
-       ifnet_t interface,
-       const struct sockaddr_dl *ll_addr)
-{
-       if (interface == NULL) return EINVAL;
-       if (ll_addr && interface->if_addrlen == 0) {
-               interface->if_addrlen = ll_addr->sdl_alen;
-       }
-       else if (ll_addr && ll_addr->sdl_alen != interface->if_addrlen) {
-               return EINVAL;
-       }
-       return dlil_if_attach_with_address(interface, ll_addr);
-}
-
-errno_t
-ifnet_detach(
-       ifnet_t interface)
+       ifnet_t ifp)
 {
-       errno_t error;
+       int     oldval;
        
-       if (interface == NULL) return EINVAL;
+       if (ifp == NULL) return EINVAL;
        
-       error = dlil_if_detach(interface);
-       if (error == DLIL_WAIT_FOR_FREE) error = 0; /* Client should always wait for detach */
+       oldval = OSDecrementAtomic((SInt32*)&ifp->if_refcnt);
+       if (oldval == 0)
+               panic("ifnet_release - refcount decremented past zero!");
        
-       return error;
+       return 0;
 }
 
 void*
@@ -301,7 +302,8 @@ ifnet_eflags(
 
 static const ifnet_offload_t offload_mask = IFNET_CSUM_IP | IFNET_CSUM_TCP |
                        IFNET_CSUM_UDP | IFNET_CSUM_FRAGMENT | IFNET_IP_FRAGMENT |
-                       IFNET_CSUM_SUM16 | IFNET_VLAN_TAGGING | IFNET_VLAN_MTU;
+                       IFNET_CSUM_SUM16 | IFNET_VLAN_TAGGING | IFNET_VLAN_MTU |
+                       IFNET_MULTIPAGES;
 
 errno_t
 ifnet_set_offload(
@@ -382,26 +384,6 @@ ifnet_get_link_mib_data_length(
        return interface == NULL ? 0 : interface->if_linkmiblen;
 }
 
-errno_t
-ifnet_attach_protocol(
-       ifnet_t interface,
-       protocol_family_t protocol,
-       const struct ifnet_attach_proto_param *proto_details)
-{
-       if (interface == NULL || protocol == 0 || proto_details == NULL)
-               return EINVAL;
-       return dlil_attach_protocol_kpi(interface, protocol, proto_details);
-}
-
-errno_t
-ifnet_detach_protocol(
-       ifnet_t interface,
-       protocol_family_t protocol)
-{
-       if (interface == NULL || protocol == 0) return EINVAL;
-       return dlil_detach_protocol(interface, protocol);
-}
-
 errno_t
 ifnet_output(
        ifnet_t interface,
@@ -424,7 +406,7 @@ ifnet_output_raw(
        protocol_family_t protocol_family,
        mbuf_t m)
 {
-       if (interface == NULL || protocol_family == 0 || m == NULL) {
+       if (interface == NULL || m == NULL) {
                if (m)
                        mbuf_freem_list(m);
                return EINVAL;
@@ -432,47 +414,6 @@ ifnet_output_raw(
        return dlil_output(interface, protocol_family, m, NULL, NULL, 1);
 }
 
-errno_t
-ifnet_input(
-       ifnet_t interface,
-       mbuf_t first_packet,
-       const struct ifnet_stat_increment_param *stats)
-{
-       mbuf_t  last_packet = first_packet;
-       
-       if (interface == NULL || first_packet == NULL) {
-               if (first_packet)
-                       mbuf_freem_list(first_packet);
-               return EINVAL;
-       }
-       
-       while (mbuf_nextpkt(last_packet) != NULL)
-               last_packet = mbuf_nextpkt(last_packet);
-       return dlil_input_with_stats(interface, first_packet, last_packet, stats);
-}
-
-errno_t
-ifnet_ioctl(
-       ifnet_t interface,
-       protocol_family_t       protocol_family,
-       u_int32_t ioctl_code,
-       void *ioctl_arg)
-{
-       if (interface == NULL || protocol_family == 0 || ioctl_code == 0)
-               return EINVAL;
-       return dlil_ioctl(protocol_family, interface,
-                                         ioctl_code, ioctl_arg);
-}
-
-errno_t
-ifnet_event(
-       ifnet_t interface,
-       struct kern_event_msg* event_ptr)
-{
-       if (interface == NULL || event_ptr == NULL) return EINVAL;
-       return dlil_event(interface, event_ptr);
-}
-
 errno_t
 ifnet_set_mtu(
        ifnet_t interface,
@@ -607,9 +548,13 @@ ifnet_stat_increment(
        ifnet_t interface,
        const struct ifnet_stat_increment_param *counts)
 {
+       struct dlil_threading_info *thread;
        if (interface == NULL) return EINVAL;
-       
-       lck_spin_lock(dlil_input_lock);
+
+               if ((thread = interface->if_input_thread) == NULL || (dlil_multithreaded_input == 0))
+               thread = dlil_lo_thread_ptr;
+
+       lck_mtx_lock(thread->input_lck);
 
        interface->if_data.ifi_ipackets += counts->packets_in;
        interface->if_data.ifi_ibytes += counts->bytes_in;
@@ -625,7 +570,7 @@ ifnet_stat_increment(
        /* Touch the last change time. */
        TOUCHLASTCHANGE(&interface->if_lastchange);
 
-       lck_spin_unlock(dlil_input_lock);
+       lck_mtx_unlock(thread->input_lck);
        
        return 0;
 }
@@ -637,9 +582,14 @@ ifnet_stat_increment_in(
        u_int32_t bytes_in,
        u_int32_t errors_in)
 {
+       struct dlil_threading_info *thread;
+
        if (interface == NULL) return EINVAL;
        
-       lck_spin_lock(dlil_input_lock);
+               if ((thread = interface->if_input_thread) == NULL || (dlil_multithreaded_input == 0))
+               thread = dlil_lo_thread_ptr;
+
+       lck_mtx_lock(thread->input_lck);
 
        interface->if_data.ifi_ipackets += packets_in;
        interface->if_data.ifi_ibytes += bytes_in;
@@ -647,7 +597,7 @@ ifnet_stat_increment_in(
 
        TOUCHLASTCHANGE(&interface->if_lastchange);
 
-       lck_spin_unlock(dlil_input_lock);
+       lck_mtx_unlock(thread->input_lck);
        
        return 0;
 }
@@ -659,9 +609,13 @@ ifnet_stat_increment_out(
        u_int32_t bytes_out,
        u_int32_t errors_out)
 {
+       struct dlil_threading_info *thread;
        if (interface == NULL) return EINVAL;
        
-       lck_spin_lock(dlil_input_lock);
+               if ((thread = interface->if_input_thread) == NULL || (dlil_multithreaded_input == 0))
+               thread = dlil_lo_thread_ptr;
+
+       lck_mtx_lock(thread->input_lck);
 
        interface->if_data.ifi_opackets += packets_out;
        interface->if_data.ifi_obytes += bytes_out;
@@ -669,7 +623,7 @@ ifnet_stat_increment_out(
 
        TOUCHLASTCHANGE(&interface->if_lastchange);
 
-       lck_spin_unlock(dlil_input_lock);
+       lck_mtx_unlock(thread->input_lck);
        
        return 0;
 }
@@ -679,9 +633,14 @@ ifnet_set_stat(
        ifnet_t interface,
        const struct ifnet_stats_param *stats)
 {
+       struct dlil_threading_info *thread;
+
        if (interface == NULL) return EINVAL;
        
-       lck_spin_lock(dlil_input_lock);
+               if ((thread = interface->if_input_thread) == NULL || (dlil_multithreaded_input == 0))
+               thread = dlil_lo_thread_ptr;
+
+       lck_mtx_lock(thread->input_lck);
 
        interface->if_data.ifi_ipackets = stats->packets_in;
        interface->if_data.ifi_ibytes = stats->bytes_in;
@@ -700,7 +659,7 @@ ifnet_set_stat(
        /* Touch the last change time. */
        TOUCHLASTCHANGE(&interface->if_lastchange);
 
-       lck_spin_unlock(dlil_input_lock);
+       lck_mtx_unlock(thread->input_lck);
        
        return 0;
 }
@@ -710,9 +669,14 @@ ifnet_stat(
        ifnet_t interface,
        struct ifnet_stats_param *stats)
 {
+       struct dlil_threading_info *thread;
+
        if (interface == NULL) return EINVAL;
        
-       lck_spin_lock(dlil_input_lock);
+               if ((thread = interface->if_input_thread) == NULL || (dlil_multithreaded_input == 0))
+               thread = dlil_lo_thread_ptr;
+
+       lck_mtx_lock(thread->input_lck);
 
        stats->packets_in = interface->if_data.ifi_ipackets;
        stats->bytes_in = interface->if_data.ifi_ibytes;
@@ -728,7 +692,7 @@ ifnet_stat(
        stats->dropped = interface->if_data.ifi_iqdrops;
        stats->no_protocol = interface->if_data.ifi_noproto;
 
-       lck_spin_unlock(dlil_input_lock);
+       lck_mtx_unlock(thread->input_lck);
        
        return 0;
 }
@@ -737,11 +701,18 @@ errno_t
 ifnet_touch_lastchange(
        ifnet_t interface)
 {
+       struct dlil_threading_info *thread;
+
        if (interface == NULL) return EINVAL;
        
-       lck_spin_lock(dlil_input_lock);
+               if ((thread = interface->if_input_thread) == NULL || (dlil_multithreaded_input == 0))
+               thread = dlil_lo_thread_ptr;
+
+       lck_mtx_lock(thread->input_lck);
+
        TOUCHLASTCHANGE(&interface->if_lastchange);
-       lck_spin_unlock(dlil_input_lock);
+
+       lck_mtx_unlock(thread->input_lck);
        
        return 0;
 }
@@ -751,11 +722,18 @@ ifnet_lastchange(
        ifnet_t interface,
        struct timeval *last_change)
 {
+       struct dlil_threading_info *thread;
+
        if (interface == NULL) return EINVAL;
        
-       lck_spin_lock(dlil_input_lock);
+               if ((thread = interface->if_input_thread) == NULL || (dlil_multithreaded_input == 0))
+               thread = dlil_lo_thread_ptr;
+
+       lck_mtx_lock(thread->input_lck);
+
        *last_change = interface->if_data.ifi_lastchange;
-       lck_spin_unlock(dlil_input_lock);
+       
+       lck_mtx_unlock(thread->input_lck);
        
 #if IF_LASTCHANGEUPTIME
        /* Crude conversion from uptime to calendar time */
@@ -770,7 +748,7 @@ ifnet_get_address_list(
        ifnet_t interface,
        ifaddr_t **addresses)
 {
-       if (interface == NULL || addresses == NULL) return EINVAL;
+       if (addresses == NULL) return EINVAL;
        return ifnet_get_address_list_family(interface, addresses, 0);
 }
 
@@ -784,7 +762,7 @@ ifnet_get_address_list_family(
        int count = 0;
        int cmax = 0;
        
-       if (interface == NULL || addresses == NULL) return EINVAL;
+       if (addresses == NULL) return EINVAL;
        *addresses = NULL;
        
        ifnet_head_lock_shared();
@@ -1086,12 +1064,12 @@ ifnet_find_by_name(
        {
                struct ifaddr *ifa = ifnet_addrs[ifp->if_index - 1];
                struct sockaddr_dl *ll_addr;
-               
+
                if (!ifa || !ifa->ifa_addr)
                        continue;
-               
+
                ll_addr = (struct sockaddr_dl *)ifa->ifa_addr;
-               
+
                if ((ifp->if_eflags & IFEF_DETACHING) == 0 &&
                        namelen == ll_addr->sdl_nlen &&
                        (strncmp(ll_addr->sdl_data, ifname, ll_addr->sdl_nlen) == 0))
@@ -1109,42 +1087,55 @@ ifnet_find_by_name(
 }
 
 errno_t
-ifnet_list_get(
-       ifnet_family_t family,
-       ifnet_t **list,
-       u_int32_t *count)
+ifnet_list_get(ifnet_family_t family, ifnet_t **list, u_int32_t *count)
+{
+       return (ifnet_list_get_common(family, FALSE, list, count));
+}
+
+__private_extern__ errno_t
+ifnet_list_get_all(ifnet_family_t family, ifnet_t **list, u_int32_t *count)
+{
+       return (ifnet_list_get_common(family, TRUE, list, count));
+}
+
+static errno_t
+ifnet_list_get_common(ifnet_family_t family, boolean_t get_all, ifnet_t **list,
+    u_int32_t *count)
 {
        struct ifnet *ifp;
        u_int32_t cmax = 0;
        *count = 0;
        errno_t result = 0;
-       
-       if (list == NULL || count == NULL) return EINVAL;
-       
+
+       if (list == NULL || count == NULL)
+               return (EINVAL);
+
        ifnet_head_lock_shared();
-       TAILQ_FOREACH(ifp, &ifnet, if_link)
-       {
-               if (ifp->if_eflags & IFEF_DETACHING) continue;
-               if (family == 0 || ifp->if_family == family)
+       TAILQ_FOREACH(ifp, &ifnet, if_link) {
+               if ((ifp->if_eflags & IFEF_DETACHING) && !get_all)
+                       continue;
+               if (family == IFNET_FAMILY_ANY || ifp->if_family == family)
                        cmax++;
        }
-       
+
        if (cmax == 0)
                result = ENXIO;
-       
+
        if (result == 0) {
-               MALLOC(*list, ifnet_t*, sizeof(ifnet_t) * (cmax + 1), M_TEMP, M_NOWAIT);
+               MALLOC(*list, ifnet_t*, sizeof(ifnet_t) * (cmax + 1),
+                   M_TEMP, M_NOWAIT);
                if (*list == NULL)
                        result = ENOMEM;
        }
 
        if (result == 0) {
-               TAILQ_FOREACH(ifp, &ifnet, if_link)
-               {
-                       if (ifp->if_eflags & IFEF_DETACHING) continue;
-                       if (*count + 1 > cmax) break;
-                       if (family == 0 || ((ifnet_family_t)ifp->if_family) == family)
-                       {
+               TAILQ_FOREACH(ifp, &ifnet, if_link) {
+                       if ((ifp->if_eflags & IFEF_DETACHING) && !get_all)
+                               continue;
+                       if (*count + 1 > cmax)
+                               break;
+                       if (family == IFNET_FAMILY_ANY ||
+                           ((ifnet_family_t)ifp->if_family) == family) {
                                (*list)[*count] = (ifnet_t)ifp;
                                ifnet_reference((*list)[*count]);
                                (*count)++;
@@ -1153,23 +1144,22 @@ ifnet_list_get(
                (*list)[*count] = NULL;
        }
        ifnet_head_done();
-       
-       return 0;
+
+       return (result);
 }
 
 void
-ifnet_list_free(
-       ifnet_t *interfaces)
+ifnet_list_free(ifnet_t *interfaces)
 {
        int i;
-       
-       if (interfaces == NULL) return;
-       
-       for (i = 0; interfaces[i]; i++)
-       {
+
+       if (interfaces == NULL)
+               return;
+
+       for (i = 0; interfaces[i]; i++) {
                ifnet_release(interfaces[i]);
        }
-       
+
        FREE(interfaces, M_TEMP);
 }