Annotation of XNU/osfmk/ipc/ipc_hash.c, revision 1.1

1.1     ! root        1: /*
        !             2:  * Copyright (c) 2000 Apple Computer, Inc. All rights reserved.
        !             3:  *
        !             4:  * @APPLE_LICENSE_HEADER_START@
        !             5:  * 
        !             6:  * The contents of this file constitute Original Code as defined in and
        !             7:  * are subject to the Apple Public Source License Version 1.1 (the
        !             8:  * "License").  You may not use this file except in compliance with the
        !             9:  * License.  Please obtain a copy of the License at
        !            10:  * http://www.apple.com/publicsource and read it before using this file.
        !            11:  * 
        !            12:  * This Original Code and all software distributed under the License are
        !            13:  * distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, EITHER
        !            14:  * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
        !            15:  * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
        !            16:  * FITNESS FOR A PARTICULAR PURPOSE OR NON-INFRINGEMENT.  Please see the
        !            17:  * License for the specific language governing rights and limitations
        !            18:  * under the License.
        !            19:  * 
        !            20:  * @APPLE_LICENSE_HEADER_END@
        !            21:  */
        !            22: /*
        !            23:  * @OSF_COPYRIGHT@
        !            24:  */
        !            25: /* 
        !            26:  * Mach Operating System
        !            27:  * Copyright (c) 1991,1990,1989 Carnegie Mellon University
        !            28:  * All Rights Reserved.
        !            29:  * 
        !            30:  * Permission to use, copy, modify and distribute this software and its
        !            31:  * documentation is hereby granted, provided that both the copyright
        !            32:  * notice and this permission notice appear in all copies of the
        !            33:  * software, derivative works or modified versions, and any portions
        !            34:  * thereof, and that both notices appear in supporting documentation.
        !            35:  * 
        !            36:  * CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS "AS IS"
        !            37:  * CONDITION.  CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND FOR
        !            38:  * ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE.
        !            39:  * 
        !            40:  * Carnegie Mellon requests users of this software to return to
        !            41:  * 
        !            42:  *  Software Distribution Coordinator  or  [email protected]
        !            43:  *  School of Computer Science
        !            44:  *  Carnegie Mellon University
        !            45:  *  Pittsburgh PA 15213-3890
        !            46:  * 
        !            47:  * any improvements or extensions that they make and grant Carnegie Mellon
        !            48:  * the rights to redistribute these changes.
        !            49:  */
        !            50: /*
        !            51:  */
        !            52: /*
        !            53:  *     File:   ipc/ipc_hash.c
        !            54:  *     Author: Rich Draves
        !            55:  *     Date:   1989
        !            56:  *
        !            57:  *     Entry hash table operations.
        !            58:  */
        !            59: 
        !            60: #include <mach/boolean.h>
        !            61: #include <mach/port.h>
        !            62: #include <kern/lock.h>
        !            63: #include <kern/kalloc.h>
        !            64: #include <ipc/port.h>
        !            65: #include <ipc/ipc_space.h>
        !            66: #include <ipc/ipc_object.h>
        !            67: #include <ipc/ipc_entry.h>
        !            68: #include <ipc/ipc_hash.h>
        !            69: #include <ipc/ipc_init.h>
        !            70: 
        !            71: #include <mach_ipc_debug.h>
        !            72: 
        !            73: #if    MACH_IPC_DEBUG
        !            74: #include <mach/kern_return.h>
        !            75: #include <mach_debug/hash_info.h>
        !            76: #include <vm/vm_map.h>
        !            77: #include <vm/vm_kern.h>
        !            78: #include <vm/vm_user.h>
        !            79: #endif /* MACH_IPC_DEBUG */
        !            80: 
        !            81: /*
        !            82:  * Forward declarations 
        !            83:  */
        !            84: 
        !            85: /* Lookup (space, obj) in global hash table */
        !            86: boolean_t ipc_hash_global_lookup(
        !            87:        ipc_space_t             space,
        !            88:        ipc_object_t            obj,
        !            89:        mach_port_name_t        *namep,
        !            90:        ipc_tree_entry_t        *entryp);
        !            91: 
        !            92: /* Insert an entry into the global reverse hash table */
        !            93: void ipc_hash_global_insert(
        !            94:        ipc_space_t             space,
        !            95:        ipc_object_t            obj,
        !            96:        mach_port_name_t        name,
        !            97:        ipc_tree_entry_t        entry);
        !            98: 
        !            99: /* Delete an entry from the local reverse hash table */
        !           100: void ipc_hash_local_delete(
        !           101:        ipc_space_t             space,
        !           102:        ipc_object_t            obj,
        !           103:        mach_port_index_t       index,
        !           104:        ipc_entry_t             entry);
        !           105: 
        !           106: /*
        !           107:  *     Routine:        ipc_hash_lookup
        !           108:  *     Purpose:
        !           109:  *             Converts (space, obj) -> (name, entry).
        !           110:  *             Returns TRUE if an entry was found.
        !           111:  *     Conditions:
        !           112:  *             The space must be locked (read or write) throughout.
        !           113:  */
        !           114: 
        !           115: boolean_t
        !           116: ipc_hash_lookup(
        !           117:        ipc_space_t             space,
        !           118:        ipc_object_t            obj,
        !           119:        mach_port_name_t        *namep,
        !           120:        ipc_entry_t             *entryp)
        !           121: {
        !           122:        boolean_t       rv;
        !           123: 
        !           124:        rv = ipc_hash_local_lookup(space, obj, namep, entryp);
        !           125:        if (!rv) {
        !           126:                assert(!is_fast_space(space) || space->is_tree_hash == 0);
        !           127:                if (space->is_tree_hash > 0)
        !           128:                        rv = ipc_hash_global_lookup(space, obj, namep,
        !           129:                                (ipc_tree_entry_t *) entryp);
        !           130:        }
        !           131:        return (rv);
        !           132: }
        !           133: 
        !           134: /*
        !           135:  *     Routine:        ipc_hash_insert
        !           136:  *     Purpose:
        !           137:  *             Inserts an entry into the appropriate reverse hash table,
        !           138:  *             so that ipc_hash_lookup will find it.
        !           139:  *     Conditions:
        !           140:  *             The space must be write-locked.
        !           141:  */
        !           142: 
        !           143: void
        !           144: ipc_hash_insert(
        !           145:        ipc_space_t             space,
        !           146:        ipc_object_t            obj,
        !           147:        mach_port_name_t        name,
        !           148:        ipc_entry_t             entry)
        !           149: {
        !           150:        mach_port_index_t index;
        !           151: 
        !           152:        index = MACH_PORT_INDEX(name);
        !           153:        if ((index < space->is_table_size) &&
        !           154:            (entry == &space->is_table[index]))
        !           155:                ipc_hash_local_insert(space, obj, index, entry);
        !           156:        else {
        !           157:                assert(!is_fast_space(space));
        !           158:                ipc_hash_global_insert(space, obj, name,
        !           159:                                       (ipc_tree_entry_t) entry);
        !           160:        }
        !           161: }
        !           162: 
        !           163: /*
        !           164:  *     Routine:        ipc_hash_delete
        !           165:  *     Purpose:
        !           166:  *             Deletes an entry from the appropriate reverse hash table.
        !           167:  *     Conditions:
        !           168:  *             The space must be write-locked.
        !           169:  */
        !           170: 
        !           171: void
        !           172: ipc_hash_delete(
        !           173:        ipc_space_t             space,
        !           174:        ipc_object_t            obj,
        !           175:        mach_port_name_t        name,
        !           176:        ipc_entry_t             entry)
        !           177: {
        !           178:        mach_port_index_t index;
        !           179: 
        !           180:        index = MACH_PORT_INDEX(name);
        !           181:        if ((index < space->is_table_size) &&
        !           182:            (entry == &space->is_table[index]))
        !           183:                ipc_hash_local_delete(space, obj, index, entry);
        !           184:        else {
        !           185:                assert(!is_fast_space(space));
        !           186:                ipc_hash_global_delete(space, obj, name,
        !           187:                                       (ipc_tree_entry_t) entry);
        !           188:        }
        !           189: }
        !           190: 
        !           191: /*
        !           192:  *     The global reverse hash table holds splay tree entries.
        !           193:  *     It is a simple open-chaining hash table with singly-linked buckets.
        !           194:  *     Each bucket is locked separately, with an exclusive lock.
        !           195:  *     Within each bucket, move-to-front is used.
        !           196:  */
        !           197: 
        !           198: typedef natural_t ipc_hash_index_t;
        !           199: 
        !           200: ipc_hash_index_t ipc_hash_global_size;
        !           201: ipc_hash_index_t ipc_hash_global_mask;
        !           202: 
        !           203: #define IH_GLOBAL_HASH(space, obj)                                     \
        !           204:        (((((ipc_hash_index_t) ((vm_offset_t)space)) >> 4) +            \
        !           205:          (((ipc_hash_index_t) ((vm_offset_t)obj)) >> 6)) &             \
        !           206:         ipc_hash_global_mask)
        !           207: 
        !           208: typedef struct ipc_hash_global_bucket {
        !           209:        decl_mutex_data(,       ihgb_lock_data)
        !           210:        ipc_tree_entry_t        ihgb_head;
        !           211: } *ipc_hash_global_bucket_t;
        !           212: 
        !           213: #define        IHGB_NULL       ((ipc_hash_global_bucket_t) 0)
        !           214: 
        !           215: #define        ihgb_lock_init(ihgb)    mutex_init(&(ihgb)->ihgb_lock_data, \
        !           216:                                           ETAP_IPC_IHGB)
        !           217: #define        ihgb_lock(ihgb)         mutex_lock(&(ihgb)->ihgb_lock_data)
        !           218: #define        ihgb_unlock(ihgb)       mutex_unlock(&(ihgb)->ihgb_lock_data)
        !           219: 
        !           220: ipc_hash_global_bucket_t ipc_hash_global_table;
        !           221: 
        !           222: /*
        !           223:  *     Routine:        ipc_hash_global_lookup
        !           224:  *     Purpose:
        !           225:  *             Converts (space, obj) -> (name, entry).
        !           226:  *             Looks in the global table, for splay tree entries.
        !           227:  *             Returns TRUE if an entry was found.
        !           228:  *     Conditions:
        !           229:  *             The space must be locked (read or write) throughout.
        !           230:  */
        !           231: 
        !           232: boolean_t
        !           233: ipc_hash_global_lookup(
        !           234:        ipc_space_t                     space,
        !           235:        ipc_object_t                    obj,
        !           236:        mach_port_name_t                *namep,
        !           237:        ipc_tree_entry_t                *entryp)
        !           238: {
        !           239:        ipc_hash_global_bucket_t bucket;
        !           240:        ipc_tree_entry_t this, *last;
        !           241: 
        !           242:        assert(space != IS_NULL);
        !           243:        assert(obj != IO_NULL);
        !           244: 
        !           245:        assert(!is_fast_space(space));
        !           246:        bucket = &ipc_hash_global_table[IH_GLOBAL_HASH(space, obj)];
        !           247:        ihgb_lock(bucket);
        !           248: 
        !           249:        if ((this = bucket->ihgb_head) != ITE_NULL) {
        !           250:                if ((this->ite_object == obj) &&
        !           251:                    (this->ite_space == space)) {
        !           252:                        /* found it at front; no need to move */
        !           253: 
        !           254:                        *namep = this->ite_name;
        !           255:                        *entryp = this;
        !           256:                } else for (last = &this->ite_next;
        !           257:                            (this = *last) != ITE_NULL;
        !           258:                            last = &this->ite_next) {
        !           259:                        if ((this->ite_object == obj) &&
        !           260:                            (this->ite_space == space)) {
        !           261:                                /* found it; move to front */
        !           262: 
        !           263:                                *last = this->ite_next;
        !           264:                                this->ite_next = bucket->ihgb_head;
        !           265:                                bucket->ihgb_head = this;
        !           266: 
        !           267:                                *namep = this->ite_name;
        !           268:                                *entryp = this;
        !           269:                                break;
        !           270:                        }
        !           271:                }
        !           272:        }
        !           273: 
        !           274:        ihgb_unlock(bucket);
        !           275:        return this != ITE_NULL;
        !           276: }
        !           277: 
        !           278: /*
        !           279:  *     Routine:        ipc_hash_global_insert
        !           280:  *     Purpose:
        !           281:  *             Inserts an entry into the global reverse hash table.
        !           282:  *     Conditions:
        !           283:  *             The space must be write-locked.
        !           284:  */
        !           285: 
        !           286: void
        !           287: ipc_hash_global_insert(
        !           288:        ipc_space_t                     space,
        !           289:        ipc_object_t                    obj,
        !           290:        mach_port_name_t                name,
        !           291:        ipc_tree_entry_t                entry)
        !           292: {
        !           293:        ipc_hash_global_bucket_t bucket;
        !           294: 
        !           295: 
        !           296:        assert(!is_fast_space(space));
        !           297: 
        !           298: 
        !           299:        assert(entry->ite_name == name);
        !           300:        assert(space != IS_NULL);
        !           301:        assert(entry->ite_space == space);
        !           302:        assert(obj != IO_NULL);
        !           303:        assert(entry->ite_object == obj);
        !           304: 
        !           305:        space->is_tree_hash++;
        !           306:        assert(space->is_tree_hash <= space->is_tree_total);
        !           307: 
        !           308:        bucket = &ipc_hash_global_table[IH_GLOBAL_HASH(space, obj)];
        !           309:        ihgb_lock(bucket);
        !           310: 
        !           311:        /* insert at front of bucket */
        !           312: 
        !           313:        entry->ite_next = bucket->ihgb_head;
        !           314:        bucket->ihgb_head = entry;
        !           315: 
        !           316:        ihgb_unlock(bucket);
        !           317: }
        !           318: 
        !           319: /*
        !           320:  *     Routine:        ipc_hash_global_delete
        !           321:  *     Purpose:
        !           322:  *             Deletes an entry from the global reverse hash table.
        !           323:  *     Conditions:
        !           324:  *             The space must be write-locked.
        !           325:  */
        !           326: 
        !           327: void
        !           328: ipc_hash_global_delete(
        !           329:        ipc_space_t                     space,
        !           330:        ipc_object_t                    obj,
        !           331:        mach_port_name_t                name,
        !           332:        ipc_tree_entry_t                entry)
        !           333: {
        !           334:        ipc_hash_global_bucket_t bucket;
        !           335:        ipc_tree_entry_t this, *last;
        !           336: 
        !           337:        assert(!is_fast_space(space));
        !           338: 
        !           339:        assert(entry->ite_name == name);
        !           340:        assert(space != IS_NULL);
        !           341:        assert(entry->ite_space == space);
        !           342:        assert(obj != IO_NULL);
        !           343:        assert(entry->ite_object == obj);
        !           344: 
        !           345:        assert(space->is_tree_hash > 0);
        !           346:        space->is_tree_hash--;
        !           347: 
        !           348:        bucket = &ipc_hash_global_table[IH_GLOBAL_HASH(space, obj)];
        !           349:        ihgb_lock(bucket);
        !           350: 
        !           351:        for (last = &bucket->ihgb_head;
        !           352:             (this = *last) != ITE_NULL;
        !           353:             last = &this->ite_next) {
        !           354:                if (this == entry) {
        !           355:                        /* found it; remove from bucket */
        !           356: 
        !           357:                        *last = this->ite_next;
        !           358:                        break;
        !           359:                }
        !           360:        }
        !           361:        assert(this != ITE_NULL);
        !           362: 
        !           363:        ihgb_unlock(bucket);
        !           364: }
        !           365: 
        !           366: /*
        !           367:  *     Each space has a local reverse hash table, which holds
        !           368:  *     entries from the space's table.  In fact, the hash table
        !           369:  *     just uses a field (ie_index) in the table itself.
        !           370:  *
        !           371:  *     The local hash table is an open-addressing hash table,
        !           372:  *     which means that when a collision occurs, instead of
        !           373:  *     throwing the entry into a bucket, the entry is rehashed
        !           374:  *     to another position in the table.  In this case the rehash
        !           375:  *     is very simple: linear probing (ie, just increment the position).
        !           376:  *     This simple rehash makes deletions tractable (they're still a pain),
        !           377:  *     but it means that collisions tend to build up into clumps.
        !           378:  *
        !           379:  *     Because at least one entry in the table (index 0) is always unused,
        !           380:  *     there will always be room in the reverse hash table.  If a table
        !           381:  *     with n slots gets completely full, the reverse hash table will
        !           382:  *     have one giant clump of n-1 slots and one free slot somewhere.
        !           383:  *     Because entries are only entered into the reverse table if they
        !           384:  *     are pure send rights (not receive, send-once, port-set,
        !           385:  *     or dead-name rights), and free entries of course aren't entered,
        !           386:  *     I expect the reverse hash table won't get unreasonably full.
        !           387:  *
        !           388:  *     Ordered hash tables (Amble & Knuth, Computer Journal, v. 17, no. 2,
        !           389:  *     pp. 135-142.) may be desirable here.  They can dramatically help
        !           390:  *     unsuccessful lookups.  But unsuccessful lookups are almost always
        !           391:  *     followed by insertions, and those slow down somewhat.  They
        !           392:  *     also can help deletions somewhat.  Successful lookups aren't affected.
        !           393:  *     So possibly a small win; probably nothing significant.
        !           394:  */
        !           395: 
        !           396: #define        IH_LOCAL_HASH(obj, size)                                \
        !           397:                ((((mach_port_index_t) (obj)) >> 6) % (size))
        !           398: 
        !           399: /*
        !           400:  *     Routine:        ipc_hash_local_lookup
        !           401:  *     Purpose:
        !           402:  *             Converts (space, obj) -> (name, entry).
        !           403:  *             Looks in the space's local table, for table entries.
        !           404:  *             Returns TRUE if an entry was found.
        !           405:  *     Conditions:
        !           406:  *             The space must be locked (read or write) throughout.
        !           407:  */
        !           408: 
        !           409: boolean_t
        !           410: ipc_hash_local_lookup(
        !           411:        ipc_space_t             space,
        !           412:        ipc_object_t            obj,
        !           413:        mach_port_name_t        *namep,
        !           414:        ipc_entry_t             *entryp)
        !           415: {
        !           416:        ipc_entry_t table;
        !           417:        ipc_entry_num_t size;
        !           418:        mach_port_index_t hindex, index;
        !           419: 
        !           420:        assert(space != IS_NULL);
        !           421:        assert(obj != IO_NULL);
        !           422: 
        !           423:        table = space->is_table;
        !           424:        size = space->is_table_size;
        !           425:        hindex = IH_LOCAL_HASH(obj, size);
        !           426: 
        !           427:        /*
        !           428:         *      Ideally, table[hindex].ie_index is the name we want.
        !           429:         *      However, must check ie_object to verify this,
        !           430:         *      because collisions can happen.  In case of a collision,
        !           431:         *      search farther along in the clump.
        !           432:         */
        !           433: 
        !           434:        while ((index = table[hindex].ie_index) != 0) {
        !           435:                ipc_entry_t entry = &table[index];
        !           436: 
        !           437:                if (entry->ie_object == obj) {
        !           438:                        *entryp = entry;
        !           439:                        *namep = MACH_PORT_MAKE(index,
        !           440:                                                IE_BITS_GEN(entry->ie_bits));
        !           441:                        return TRUE;
        !           442:                }
        !           443: 
        !           444:                if (++hindex == size)
        !           445:                        hindex = 0;
        !           446:        }
        !           447: 
        !           448:        return FALSE;
        !           449: }
        !           450: 
        !           451: /*
        !           452:  *     Routine:        ipc_hash_local_insert
        !           453:  *     Purpose:
        !           454:  *             Inserts an entry into the space's reverse hash table.
        !           455:  *     Conditions:
        !           456:  *             The space must be write-locked.
        !           457:  */
        !           458: 
        !           459: void
        !           460: ipc_hash_local_insert(
        !           461:        ipc_space_t             space,
        !           462:        ipc_object_t            obj,
        !           463:        mach_port_index_t       index,
        !           464:        ipc_entry_t             entry)
        !           465: {
        !           466:        ipc_entry_t table;
        !           467:        ipc_entry_num_t size;
        !           468:        mach_port_index_t hindex;
        !           469: 
        !           470:        assert(index != 0);
        !           471:        assert(space != IS_NULL);
        !           472:        assert(obj != IO_NULL);
        !           473: 
        !           474:        table = space->is_table;
        !           475:        size = space->is_table_size;
        !           476:        hindex = IH_LOCAL_HASH(obj, size);
        !           477: 
        !           478:        assert(entry == &table[index]);
        !           479:        assert(entry->ie_object == obj);
        !           480: 
        !           481:        /*
        !           482:         *      We want to insert at hindex, but there may be collisions.
        !           483:         *      If a collision occurs, search for the end of the clump
        !           484:         *      and insert there.
        !           485:         */
        !           486: 
        !           487:        while (table[hindex].ie_index != 0) {
        !           488:                if (++hindex == size)
        !           489:                        hindex = 0;
        !           490:        }
        !           491: 
        !           492:        table[hindex].ie_index = index;
        !           493: }
        !           494: 
        !           495: /*
        !           496:  *     Routine:        ipc_hash_local_delete
        !           497:  *     Purpose:
        !           498:  *             Deletes an entry from the space's reverse hash table.
        !           499:  *     Conditions:
        !           500:  *             The space must be write-locked.
        !           501:  */
        !           502: 
        !           503: void
        !           504: ipc_hash_local_delete(
        !           505:        ipc_space_t             space,
        !           506:        ipc_object_t            obj,
        !           507:        mach_port_index_t       index,
        !           508:        ipc_entry_t             entry)
        !           509: {
        !           510:        ipc_entry_t table;
        !           511:        ipc_entry_num_t size;
        !           512:        mach_port_index_t hindex, dindex;
        !           513: 
        !           514:        assert(index != MACH_PORT_NULL);
        !           515:        assert(space != IS_NULL);
        !           516:        assert(obj != IO_NULL);
        !           517: 
        !           518:        table = space->is_table;
        !           519:        size = space->is_table_size;
        !           520:        hindex = IH_LOCAL_HASH(obj, size);
        !           521: 
        !           522:        assert(entry == &table[index]);
        !           523:        assert(entry->ie_object == obj);
        !           524: 
        !           525:        /*
        !           526:         *      First check we have the right hindex for this index.
        !           527:         *      In case of collision, we have to search farther
        !           528:         *      along in this clump.
        !           529:         */
        !           530: 
        !           531:        while (table[hindex].ie_index != index) {
        !           532:                if (++hindex == size)
        !           533:                        hindex = 0;
        !           534:        }
        !           535: 
        !           536:        /*
        !           537:         *      Now we want to set table[hindex].ie_index = 0.
        !           538:         *      But if we aren't the last index in a clump,
        !           539:         *      this might cause problems for lookups of objects
        !           540:         *      farther along in the clump that are displaced
        !           541:         *      due to collisions.  Searches for them would fail
        !           542:         *      at hindex instead of succeeding.
        !           543:         *
        !           544:         *      So we must check the clump after hindex for objects
        !           545:         *      that are so displaced, and move one up to the new hole.
        !           546:         *
        !           547:         *              hindex - index of new hole in the clump
        !           548:         *              dindex - index we are checking for a displaced object
        !           549:         *
        !           550:         *      When we move a displaced object up into the hole,
        !           551:         *      it creates a new hole, and we have to repeat the process
        !           552:         *      until we get to the end of the clump.
        !           553:         */
        !           554: 
        !           555:        for (dindex = hindex; index != 0; hindex = dindex) {
        !           556:                for (;;) {
        !           557:                        mach_port_index_t tindex;
        !           558:                        ipc_object_t tobj;
        !           559: 
        !           560:                        if (++dindex == size)
        !           561:                                dindex = 0;
        !           562:                        assert(dindex != hindex);
        !           563: 
        !           564:                        /* are we at the end of the clump? */
        !           565: 
        !           566:                        index = table[dindex].ie_index;
        !           567:                        if (index == 0)
        !           568:                                break;
        !           569: 
        !           570:                        /* is this a displaced object? */
        !           571: 
        !           572:                        tobj = table[index].ie_object;
        !           573:                        assert(tobj != IO_NULL);
        !           574:                        tindex = IH_LOCAL_HASH(tobj, size);
        !           575: 
        !           576:                        if ((dindex < hindex) ?
        !           577:                            ((dindex < tindex) && (tindex <= hindex)) :
        !           578:                            ((dindex < tindex) || (tindex <= hindex)))
        !           579:                                break;
        !           580:                }
        !           581: 
        !           582:                table[hindex].ie_index = index;
        !           583:        }
        !           584: }
        !           585: 
        !           586: /*
        !           587:  *     Routine:        ipc_hash_init
        !           588:  *     Purpose:
        !           589:  *             Initialize the reverse hash table implementation.
        !           590:  */
        !           591: 
        !           592: void
        !           593: ipc_hash_init(void)
        !           594: {
        !           595:        ipc_hash_index_t i;
        !           596: 
        !           597:        /* if not configured, initialize ipc_hash_global_size */
        !           598: 
        !           599:        if (ipc_hash_global_size == 0) {
        !           600:                ipc_hash_global_size = ipc_tree_entry_max >> 8;
        !           601:                if (ipc_hash_global_size < 32)
        !           602:                        ipc_hash_global_size = 32;
        !           603:        }
        !           604: 
        !           605:        /* make sure it is a power of two */
        !           606: 
        !           607:        ipc_hash_global_mask = ipc_hash_global_size - 1;
        !           608:        if ((ipc_hash_global_size & ipc_hash_global_mask) != 0) {
        !           609:                natural_t bit;
        !           610: 
        !           611:                /* round up to closest power of two */
        !           612: 
        !           613:                for (bit = 1;; bit <<= 1) {
        !           614:                        ipc_hash_global_mask |= bit;
        !           615:                        ipc_hash_global_size = ipc_hash_global_mask + 1;
        !           616: 
        !           617:                        if ((ipc_hash_global_size & ipc_hash_global_mask) == 0)
        !           618:                                break;
        !           619:                }
        !           620:        }
        !           621: 
        !           622:        /* allocate ipc_hash_global_table */
        !           623: 
        !           624:        ipc_hash_global_table = (ipc_hash_global_bucket_t)
        !           625:                kalloc((vm_size_t) (ipc_hash_global_size *
        !           626:                                    sizeof(struct ipc_hash_global_bucket)));
        !           627:        assert(ipc_hash_global_table != IHGB_NULL);
        !           628: 
        !           629:        /* and initialize it */
        !           630: 
        !           631:        for (i = 0; i < ipc_hash_global_size; i++) {
        !           632:                ipc_hash_global_bucket_t bucket;
        !           633: 
        !           634:                bucket = &ipc_hash_global_table[i];
        !           635:                ihgb_lock_init(bucket);
        !           636:                bucket->ihgb_head = ITE_NULL;
        !           637:        }
        !           638: }
        !           639: 
        !           640: #if    MACH_IPC_DEBUG
        !           641: 
        !           642: /*
        !           643:  *     Routine:        ipc_hash_info
        !           644:  *     Purpose:
        !           645:  *             Return information about the global reverse hash table.
        !           646:  *             Fills the buffer with as much information as possible
        !           647:  *             and returns the desired size of the buffer.
        !           648:  *     Conditions:
        !           649:  *             Nothing locked.  The caller should provide
        !           650:  *             possibly-pageable memory.
        !           651:  */
        !           652: 
        !           653: 
        !           654: ipc_hash_index_t
        !           655: ipc_hash_info(
        !           656:        hash_info_bucket_t      *info,
        !           657:        mach_msg_type_number_t count)
        !           658: {
        !           659:        ipc_hash_index_t i;
        !           660: 
        !           661:        if (ipc_hash_global_size < count)
        !           662:                count = ipc_hash_global_size;
        !           663: 
        !           664:        for (i = 0; i < count; i++) {
        !           665:                ipc_hash_global_bucket_t bucket = &ipc_hash_global_table[i];
        !           666:                unsigned int bucket_count = 0;
        !           667:                ipc_tree_entry_t entry;
        !           668: 
        !           669:                ihgb_lock(bucket);
        !           670:                for (entry = bucket->ihgb_head;
        !           671:                     entry != ITE_NULL;
        !           672:                     entry = entry->ite_next)
        !           673:                        bucket_count++;
        !           674:                ihgb_unlock(bucket);
        !           675: 
        !           676:                /* don't touch pageable memory while holding locks */
        !           677:                info[i].hib_count = bucket_count;
        !           678:        }
        !           679: 
        !           680:        return ipc_hash_global_size;
        !           681: }
        !           682: 
        !           683: #endif /* MACH_IPC_DEBUG */

unix.superglobalmegacorp.com

This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.