]> git.saurik.com Git - apple/xnu.git/blobdiff - osfmk/ipc/ipc_space.c
xnu-2782.10.72.tar.gz
[apple/xnu.git] / osfmk / ipc / ipc_space.c
index c88fbe0de1b832d6e4d4a3860a26275148c4ca4c..7d0305e32e2e144895d2c7a2e6bd4ca8aee91160 100644 (file)
@@ -1,23 +1,29 @@
 /*
  * Copyright (c) 2000-2004 Apple Computer, Inc. All rights reserved.
  *
 /*
  * Copyright (c) 2000-2004 Apple Computer, 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@
  */
 /*
  * @OSF_COPYRIGHT@
  */
 /*
  * @OSF_COPYRIGHT@
  * any improvements or extensions that they make and grant Carnegie Mellon
  * the rights to redistribute these changes.
  */
  * any improvements or extensions that they make and grant Carnegie Mellon
  * the rights to redistribute these changes.
  */
+/*
+ * NOTICE: This file was modified by McAfee Research in 2004 to introduce
+ * support for mandatory and extensible security protections.  This notice
+ * is included in support of clause 2.2 (b) of the Apple Public License,
+ * Version 2.0.
+ */
 /*
  */
 /*
 /*
  */
 /*
@@ -57,8 +69,6 @@
  *     Functions to manipulate IPC capability spaces.
  */
 
  *     Functions to manipulate IPC capability spaces.
  */
 
-#include <mach_kdb.h>
-
 #include <mach/boolean.h>
 #include <mach/kern_return.h>
 #include <mach/port.h>
 #include <mach/boolean.h>
 #include <mach/kern_return.h>
 #include <mach/port.h>
@@ -67,7 +77,6 @@
 #include <kern/zalloc.h>
 #include <ipc/port.h>
 #include <ipc/ipc_entry.h>
 #include <kern/zalloc.h>
 #include <ipc/port.h>
 #include <ipc/ipc_entry.h>
-#include <ipc/ipc_splay.h>
 #include <ipc/ipc_object.h>
 #include <ipc/ipc_hash.h>
 #include <ipc/ipc_table.h>
 #include <ipc/ipc_object.h>
 #include <ipc/ipc_hash.h>
 #include <ipc/ipc_table.h>
 zone_t ipc_space_zone;
 ipc_space_t ipc_space_kernel;
 ipc_space_t ipc_space_reply;
 zone_t ipc_space_zone;
 ipc_space_t ipc_space_kernel;
 ipc_space_t ipc_space_reply;
-#if    MACH_KDB
-ipc_space_t default_pager_space;
-#endif /* MACH_KDB */
 
 /*
  *     Routine:        ipc_space_reference
  *     Routine:        ipc_space_release
  *     Purpose:
 
 /*
  *     Routine:        ipc_space_reference
  *     Routine:        ipc_space_release
  *     Purpose:
- *             Function versions of the IPC space macros.
- *             The "is_" cover macros can be defined to use the
- *             macros or the functions, as desired.
+ *             Function versions of the IPC space inline reference.
  */
 
 void
 ipc_space_reference(
        ipc_space_t     space)
 {
  */
 
 void
 ipc_space_reference(
        ipc_space_t     space)
 {
-       ipc_space_reference_macro(space);
+       is_reference(space);
 }
 
 void
 ipc_space_release(
        ipc_space_t     space)
 {
 }
 
 void
 ipc_space_release(
        ipc_space_t     space)
 {
-       ipc_space_release_macro(space);
+       is_release(space);
 }
 
 /*
 }
 
 /*
@@ -157,20 +161,15 @@ ipc_space_create(
        }
        table[new_size-1].ie_next = 0;
 
        }
        table[new_size-1].ie_next = 0;
 
-       is_ref_lock_init(space);
-       space->is_references = 2;
-
        is_lock_init(space);
        is_lock_init(space);
-       space->is_active = TRUE;
-       space->is_growing = FALSE;
-       space->is_table = table;
+       space->is_bits = 2; /* 2 refs, active, not growing */
        space->is_table_size = new_size;
        space->is_table_size = new_size;
+       space->is_table_free = new_size - 1;
+       space->is_table = table;
        space->is_table_next = initial+1;
        space->is_table_next = initial+1;
-
-       ipc_splay_tree_init(&space->is_tree);
-       space->is_tree_total = 0;
-       space->is_tree_small = 0;
-       space->is_tree_hash = 0;
+       space->is_task = NULL;
+       space->is_low_mod = new_size;
+       space->is_high_mod = 0;
 
        *spacep = space;
        return KERN_SUCCESS;
 
        *spacep = space;
        return KERN_SUCCESS;
@@ -201,11 +200,14 @@ ipc_space_create_special(
        if (space == IS_NULL)
                return KERN_RESOURCE_SHORTAGE;
 
        if (space == IS_NULL)
                return KERN_RESOURCE_SHORTAGE;
 
-       is_ref_lock_init(space);
-       space->is_references = 1;
-
        is_lock_init(space);
        is_lock_init(space);
-       space->is_active = FALSE;
+
+       space->is_bits       = IS_INACTIVE | 1; /* 1 ref, not active, not growing */
+       space->is_table      = IE_NULL;
+       space->is_task       = TASK_NULL;
+       space->is_table_next = 0;
+       space->is_low_mod    = 0;
+       space->is_high_mod   = 0;
 
        *spacep = space;
        return KERN_SUCCESS;
 
        *spacep = space;
        return KERN_SUCCESS;
@@ -222,7 +224,6 @@ void
 ipc_space_clean(
        ipc_space_t space)
 {
 ipc_space_clean(
        ipc_space_t space)
 {
-       ipc_tree_entry_t tentry;
        ipc_entry_t table;
        ipc_entry_num_t size;
        mach_port_index_t index;
        ipc_entry_t table;
        ipc_entry_num_t size;
        mach_port_index_t index;
@@ -232,17 +233,19 @@ ipc_space_clean(
         *      we must wait until they finish and figure
         *      out the space died.
         */
         *      we must wait until they finish and figure
         *      out the space died.
         */
+ retry:
        is_write_lock(space);
        is_write_lock(space);
-       while (space->is_growing)
+       while (is_growing(space))
                is_write_sleep(space);
 
                is_write_sleep(space);
 
+       if (!is_active(space)) {
+               is_write_unlock(space);
+               return;
+       }
+
        /*
         *      Now we can futz with it since we have the write lock.
         */
        /*
         *      Now we can futz with it since we have the write lock.
         */
-#if    MACH_KDB
-       if (space == default_pager_space)
-               default_pager_space = IS_NULL;
-#endif /* MACH_KDB */
 
        table = space->is_table;
        size = space->is_table_size;
 
        table = space->is_table;
        size = space->is_table_size;
@@ -255,40 +258,23 @@ ipc_space_clean(
                if (type != MACH_PORT_TYPE_NONE) {
                        mach_port_name_t name = MACH_PORT_MAKE(index,
                                                IE_BITS_GEN(entry->ie_bits));
                if (type != MACH_PORT_TYPE_NONE) {
                        mach_port_name_t name = MACH_PORT_MAKE(index,
                                                IE_BITS_GEN(entry->ie_bits));
-                       ipc_right_destroy(space, name, entry);
+                       ipc_right_destroy(space, name, entry, FALSE, 0); /* unlocks space */
+                       goto retry;
                }
        }
 
                }
        }
 
-       /*
+        /*
         * JMM - Now the table is cleaned out.  We don't bother shrinking the
         * size of the table at this point, but we probably should if it is
         * JMM - Now the table is cleaned out.  We don't bother shrinking the
         * size of the table at this point, but we probably should if it is
-        * really large.  Lets just clean up the splay tree.
+        * really large.
         */
         */
- start_splay:
-       for (tentry = ipc_splay_traverse_start(&space->is_tree);
-            tentry != ITE_NULL;
-            tentry = ipc_splay_traverse_next(&space->is_tree, TRUE)) {
-               mach_port_type_t type;
-               mach_port_name_t name = tentry->ite_name;
-
-               type = IE_BITS_TYPE(tentry->ite_bits);
-               /*
-                * If it is a real right, then destroy it.  This will have the
-                * side effect of removing it from the splay, so start over.
-                */
-               if(type != MACH_PORT_TYPE_NONE) {
-                       ipc_splay_traverse_finish(&space->is_tree);
-                       ipc_right_destroy(space, name, &tentry->ite_entry);
-                       goto start_splay;
-               }
-       }
-       ipc_splay_traverse_finish(&space->is_tree);
+       
        is_write_unlock(space);
 }
 
 
 /*
        is_write_unlock(space);
 }
 
 
 /*
- *     Routine:        ipc_space_destroy
+ *     Routine:        ipc_space_terminate
  *     Purpose:
  *             Marks the space as dead and cleans up the entries.
  *             Does nothing if the space is already dead.
  *     Purpose:
  *             Marks the space as dead and cleans up the entries.
  *             Does nothing if the space is already dead.
@@ -297,11 +283,9 @@ ipc_space_clean(
  */
 
 void
  */
 
 void
-ipc_space_destroy(
+ipc_space_terminate(
        ipc_space_t     space)
 {
        ipc_space_t     space)
 {
-       boolean_t active;
-       ipc_tree_entry_t tentry;
        ipc_entry_t table;
        ipc_entry_num_t size;
        mach_port_index_t index;
        ipc_entry_t table;
        ipc_entry_num_t size;
        mach_port_index_t index;
@@ -309,31 +293,26 @@ ipc_space_destroy(
        assert(space != IS_NULL);
 
        is_write_lock(space);
        assert(space != IS_NULL);
 
        is_write_lock(space);
-       active = space->is_active;
-       space->is_active = FALSE;
-       is_write_unlock(space);
-
-       if (!active)
+       if (!is_active(space)) {
+               is_write_unlock(space);
                return;
                return;
-
+       }
+       is_mark_inactive(space);
 
        /*
         *      If somebody is trying to grow the table,
         *      we must wait until they finish and figure
         *      out the space died.
         */
 
        /*
         *      If somebody is trying to grow the table,
         *      we must wait until they finish and figure
         *      out the space died.
         */
-       is_read_lock(space);
-       while (space->is_growing)
-               is_read_sleep(space);
+       while (is_growing(space))
+               is_write_sleep(space);
+
+       is_write_unlock(space);
+
 
 
-       is_read_unlock(space);
        /*
         *      Now we can futz with it unlocked.
         */
        /*
         *      Now we can futz with it unlocked.
         */
-#if    MACH_KDB
-       if (space == default_pager_space)
-               default_pager_space = IS_NULL;
-#endif /* MACH_KDB */
 
        table = space->is_table;
        size = space->is_table_size;
 
        table = space->is_table;
        size = space->is_table_size;
@@ -348,29 +327,13 @@ ipc_space_destroy(
 
                        name = MACH_PORT_MAKE(index,
                                              IE_BITS_GEN(entry->ie_bits));
 
                        name = MACH_PORT_MAKE(index,
                                              IE_BITS_GEN(entry->ie_bits));
-                       ipc_right_clean(space, name, entry);
+                       ipc_right_terminate(space, name, entry);
                }
        }
 
        it_entries_free(space->is_table_next-1, table);
        space->is_table_size = 0;
                }
        }
 
        it_entries_free(space->is_table_next-1, table);
        space->is_table_size = 0;
-
-       for (tentry = ipc_splay_traverse_start(&space->is_tree);
-            tentry != ITE_NULL;
-            tentry = ipc_splay_traverse_next(&space->is_tree, TRUE)) {
-               mach_port_type_t type;
-               mach_port_name_t name = tentry->ite_name;
-
-               type = IE_BITS_TYPE(tentry->ite_bits);
-               assert(type != MACH_PORT_TYPE_NONE);
-
-               ipc_right_clean(space, name, &tentry->ite_entry);
-
-               if(type == MACH_PORT_TYPE_SEND)
-                       ipc_hash_global_delete(space, tentry->ite_object,
-                                              name, tentry);
-       }
-       ipc_splay_traverse_finish(&space->is_tree);
+       space->is_table_free = 0;
 
        /*
         *      Because the space is now dead,
 
        /*
         *      Because the space is now dead,