]> git.saurik.com Git - apple/xnu.git/blobdiff - bsd/netinet/ip_flow.c
xnu-1228.12.14.tar.gz
[apple/xnu.git] / bsd / netinet / ip_flow.c
index ba3ca98163ea6e873582bf70990d7c0f18de0694..4fb3f8596b3a9caf6ef176203a0adaa98c6d420d 100644 (file)
@@ -1,23 +1,29 @@
 /*
 /*
- * Copyright (c) 2000 Apple Computer, Inc. All rights reserved.
+ * Copyright (c) 2000,2007 Apple Inc. All rights reserved.
  *
  *
- * @APPLE_LICENSE_HEADER_START@
+ * @APPLE_OSREFERENCE_LICENSE_HEADER_START@
  * 
  * 
- * The contents of this file constitute Original Code as defined in and
- * are subject to the Apple Public Source License Version 1.1 (the
- * "License").  You may not use this file except in compliance with the
- * License.  Please obtain a copy of the License at
- * http://www.apple.com/publicsource and read it before using this file.
+ * 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.
  * 
  * 
- * This Original Code and all software distributed under the License are
- * distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, EITHER
+ * 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,
  * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
  * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
- * FITNESS FOR A PARTICULAR PURPOSE OR NON-INFRINGEMENT.  Please see the
- * License for the specific language governing rights and limitations
- * under the License.
+ * 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_HEADER_END@
+ * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
  */
 /*-
  * Copyright (c) 1998 The NetBSD Foundation, Inc.
  */
 /*-
  * Copyright (c) 1998 The NetBSD Foundation, Inc.
@@ -66,6 +72,7 @@
 #include <sys/kernel.h>
 
 #include <sys/sysctl.h>
 #include <sys/kernel.h>
 
 #include <sys/sysctl.h>
+#include <libkern/OSAtomic.h>
 
 #include <net/if.h>
 #include <net/route.h>
 
 #include <net/if.h>
 #include <net/route.h>
@@ -212,24 +219,21 @@ ipflow_addstats(
        struct ipflow *ipf)
 {
        ipf->ipf_ro.ro_rt->rt_use += ipf->ipf_uses;
        struct ipflow *ipf)
 {
        ipf->ipf_ro.ro_rt->rt_use += ipf->ipf_uses;
-       ipstat.ips_cantforward += ipf->ipf_errors + ipf->ipf_dropped;
-       ipstat.ips_forward += ipf->ipf_uses;
-       ipstat.ips_fastforward += ipf->ipf_uses;
+       OSAddAtomic(ipf->ipf_errors + ipf->ipf_dropped, (SInt32*)&ipstat.ips_cantforward);
+       OSAddAtomic(ipf->ipf_uses, (SInt32*)&ipstat.ips_forward);
+       OSAddAtomic(ipf->ipf_uses, (SInt32*)&ipstat.ips_fastforward);
 }
 
 static void
 ipflow_free(
        struct ipflow *ipf)
 {
 }
 
 static void
 ipflow_free(
        struct ipflow *ipf)
 {
-       int s;
        /*
         * Remove the flow from the hash table (at elevated IPL).
         * Once it's off the list, we can deal with it at normal
         * network IPL.
         */
        /*
         * Remove the flow from the hash table (at elevated IPL).
         * Once it's off the list, we can deal with it at normal
         * network IPL.
         */
-       s = splimp();
        LIST_REMOVE(ipf, ipf_next);
        LIST_REMOVE(ipf, ipf_next);
-       splx(s);
        ipflow_addstats(ipf);
        rtfree(ipf->ipf_ro.ro_rt);
        ipflow_inuse--;
        ipflow_addstats(ipf);
        rtfree(ipf->ipf_ro.ro_rt);
        ipflow_inuse--;
@@ -242,7 +246,6 @@ ipflow_reap(
 {
        struct ipflow *ipf, *maybe_ipf = NULL;
        int idx;
 {
        struct ipflow *ipf, *maybe_ipf = NULL;
        int idx;
-       int s;
 
        for (idx = 0; idx < IPFLOW_HASHSIZE; idx++) {
                ipf = LIST_FIRST(&ipflows[idx]);
 
        for (idx = 0; idx < IPFLOW_HASHSIZE; idx++) {
                ipf = LIST_FIRST(&ipflows[idx]);
@@ -273,14 +276,13 @@ ipflow_reap(
        /*
         * Remove the entry from the flow table.
         */
        /*
         * Remove the entry from the flow table.
         */
-       s = splimp();
        LIST_REMOVE(ipf, ipf_next);
        LIST_REMOVE(ipf, ipf_next);
-       splx(s);
        ipflow_addstats(ipf);
        rtfree(ipf->ipf_ro.ro_rt);
        ipflow_addstats(ipf);
        rtfree(ipf->ipf_ro.ro_rt);
+       ipf->ipf_ro.ro_rt = NULL;
        return ipf;
 }
        return ipf;
 }
-
+/* note: called under the ip_mutex lock */
 void
 ipflow_slowtimo(
        void)
 void
 ipflow_slowtimo(
        void)
@@ -297,6 +299,8 @@ ipflow_slowtimo(
                        } else {
                                ipf->ipf_last_uses = ipf->ipf_uses;
                                ipf->ipf_ro.ro_rt->rt_use += ipf->ipf_uses;
                        } else {
                                ipf->ipf_last_uses = ipf->ipf_uses;
                                ipf->ipf_ro.ro_rt->rt_use += ipf->ipf_uses;
+                               OSAddAtomic(ipf->ipf_uses, (SInt32*)&ipstat.ips_forward);
+                               OSAddAtomic(ipf->ipf_uses, (SInt32*)&ipstat.ips_fastforward);
                                ipstat.ips_forward += ipf->ipf_uses;
                                ipstat.ips_fastforward += ipf->ipf_uses;
                                ipf->ipf_uses = 0;
                                ipstat.ips_forward += ipf->ipf_uses;
                                ipstat.ips_fastforward += ipf->ipf_uses;
                                ipf->ipf_uses = 0;
@@ -314,7 +318,6 @@ ipflow_create(
        const struct ip *const ip = mtod(m, struct ip *);
        struct ipflow *ipf;
        unsigned hash;
        const struct ip *const ip = mtod(m, struct ip *);
        struct ipflow *ipf;
        unsigned hash;
-       int s;
 
        /*
         * Don't create cache entries for ICMP messages.
 
        /*
         * Don't create cache entries for ICMP messages.
@@ -339,11 +342,10 @@ ipflow_create(
                }
                bzero((caddr_t) ipf, sizeof(*ipf));
        } else {
                }
                bzero((caddr_t) ipf, sizeof(*ipf));
        } else {
-               s = splimp();
                LIST_REMOVE(ipf, ipf_next);
                LIST_REMOVE(ipf, ipf_next);
-               splx(s);
                ipflow_addstats(ipf);
                rtfree(ipf->ipf_ro.ro_rt);
                ipflow_addstats(ipf);
                rtfree(ipf->ipf_ro.ro_rt);
+               ipf->ipf_ro.ro_rt = NULL;
                ipf->ipf_uses = ipf->ipf_last_uses = 0;
                ipf->ipf_errors = ipf->ipf_dropped = 0;
        }
                ipf->ipf_uses = ipf->ipf_last_uses = 0;
                ipf->ipf_errors = ipf->ipf_dropped = 0;
        }
@@ -351,8 +353,10 @@ ipflow_create(
        /*
         * Fill in the updated information.
         */
        /*
         * Fill in the updated information.
         */
+       lck_mtx_lock(rt_mtx);
        ipf->ipf_ro = *ro;
        ipf->ipf_ro = *ro;
-       rtref(ro->ro_rt);               //### LD 5/25/04 needs rt_mtx lock
+       rtref(ro->ro_rt);
+       lck_mtx_unlock(rt_mtx);
        ipf->ipf_dst = ip->ip_dst;
        ipf->ipf_src = ip->ip_src;
        ipf->ipf_tos = ip->ip_tos;
        ipf->ipf_dst = ip->ip_dst;
        ipf->ipf_src = ip->ip_src;
        ipf->ipf_tos = ip->ip_tos;
@@ -361,7 +365,5 @@ ipflow_create(
         * Insert into the approriate bucket of the flow table.
         */
        hash = ipflow_hash(ip->ip_dst, ip->ip_src, ip->ip_tos);
         * Insert into the approriate bucket of the flow table.
         */
        hash = ipflow_hash(ip->ip_dst, ip->ip_src, ip->ip_tos);
-       s = splimp();
        LIST_INSERT_HEAD(&ipflows[hash], ipf, ipf_next);
        LIST_INSERT_HEAD(&ipflows[hash], ipf, ipf_next);
-       splx(s);
 }
 }