]> git.saurik.com Git - apple/xnu.git/blobdiff - iokit/Kernel/IOCatalogue.cpp
xnu-6153.81.5.tar.gz
[apple/xnu.git] / iokit / Kernel / IOCatalogue.cpp
index eb8dfbafbbb8abc9cd68ff1b0aa54d90f9aeac8e..7c0201e4df7c2381f379c5e62bde771231810414 100644 (file)
@@ -2,7 +2,7 @@
  * Copyright (c) 1998-2012 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
  * 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,
  * 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@
  */
 /*
- * Copyright (c) 1998 Apple Inc.  All rights reserved. 
+ * Copyright (c) 1998 Apple Inc.  All rights reserved.
  *
  * HISTORY
  *
@@ -68,7 +68,8 @@ IOCatalogue    * gIOCatalogue;
 const OSSymbol * gIOClassKey;
 const OSSymbol * gIOProbeScoreKey;
 const OSSymbol * gIOModuleIdentifierKey;
-IORWLock         * gIOCatalogLock;
+const OSSymbol * gIOModuleIdentifierKernelKey;
+IORWLock       * gIOCatalogLock;
 
 #if PRAGMA_MARK
 #pragma mark Utility functions
@@ -83,149 +84,168 @@ IORWLock         * gIOCatalogLock;
 #define super OSObject
 OSDefineMetaClassAndStructors(IOCatalogue, OSObject)
 
+static bool isModuleLoadedNoOSKextLock(OSDictionary *theKexts,
+    OSDictionary *theModuleDict);
+
+
 /*********************************************************************
 *********************************************************************/
-void IOCatalogue::initialize(void)
+void
+IOCatalogue::initialize(void)
 {
-    OSArray              * array;
-    OSString             * errorString;
-    bool                  rc;
+       OSArray              * array;
+       OSString             * errorString;
+       bool                   rc;
 
-    extern const char * gIOKernelConfigTables;
+       extern const char * gIOKernelConfigTables;
 
-    array = OSDynamicCast(OSArray, OSUnserialize(gIOKernelConfigTables, &errorString));
-    if (!array && errorString) {
-        IOLog("KernelConfigTables syntax error: %s\n",
-            errorString->getCStringNoCopy());
-        errorString->release();
-    }
+       array = OSDynamicCast(OSArray, OSUnserialize(gIOKernelConfigTables, &errorString));
+       if (!array && errorString) {
+               IOLog("KernelConfigTables syntax error: %s\n",
+                   errorString->getCStringNoCopy());
+               errorString->release();
+       }
+
+       gIOClassKey                  = OSSymbol::withCStringNoCopy( kIOClassKey );
+       gIOProbeScoreKey             = OSSymbol::withCStringNoCopy( kIOProbeScoreKey );
+       gIOModuleIdentifierKey       = OSSymbol::withCStringNoCopy( kCFBundleIdentifierKey );
+       gIOModuleIdentifierKernelKey = OSSymbol::withCStringNoCopy( kCFBundleIdentifierKernelKey );
 
-    gIOClassKey              = OSSymbol::withCStringNoCopy( kIOClassKey );
-    gIOProbeScoreKey        = OSSymbol::withCStringNoCopy( kIOProbeScoreKey );
-    gIOModuleIdentifierKey   = OSSymbol::withCStringNoCopy( kCFBundleIdentifierKey );
 
-    assert( array && gIOClassKey && gIOProbeScoreKey 
+       assert( array && gIOClassKey && gIOProbeScoreKey
            && gIOModuleIdentifierKey);
 
-    gIOCatalogue = new IOCatalogue;
-    assert(gIOCatalogue);
-    rc = gIOCatalogue->init(array);
-    assert(rc);
-    array->release();
+       gIOCatalogue = new IOCatalogue;
+       assert(gIOCatalogue);
+       rc = gIOCatalogue->init(array);
+       assert(rc);
+       array->release();
 }
 
 /*********************************************************************
 * Initialize the IOCatalog object.
 *********************************************************************/
-OSArray * IOCatalogue::arrayForPersonality(OSDictionary * dict)
+OSArray *
+IOCatalogue::arrayForPersonality(OSDictionary * dict)
 {
-    const OSSymbol * sym;
+       const OSSymbol * sym;
 
-    sym = OSDynamicCast(OSSymbol, dict->getObject(gIOProviderClassKey));
-    if (!sym)  return (0);
+       sym = OSDynamicCast(OSSymbol, dict->getObject(gIOProviderClassKey));
+       if (!sym) {
+               return NULL;
+       }
 
-    return ((OSArray *) personalities->getObject(sym));
+       return (OSArray *) personalities->getObject(sym);
 }
 
-void IOCatalogue::addPersonality(OSDictionary * dict)
+void
+IOCatalogue::addPersonality(OSDictionary * dict)
 {
-    const OSSymbol * sym;
-    OSArray * arr;
-
-    sym = OSDynamicCast(OSSymbol, dict->getObject(gIOProviderClassKey));
-    if (!sym) return;
-    arr = (OSArray *) personalities->getObject(sym);
-    if (arr) arr->setObject(dict);
-    else
-    {
-        arr = OSArray::withObjects((const OSObject **)&dict, 1, 2);
-        personalities->setObject(sym, arr);
-        arr->release();
-    }
+       const OSSymbol * sym;
+       OSArray * arr;
+
+       sym = OSDynamicCast(OSSymbol, dict->getObject(gIOProviderClassKey));
+       if (!sym) {
+               return;
+       }
+       arr = (OSArray *) personalities->getObject(sym);
+       if (arr) {
+               arr->setObject(dict);
+       } else {
+               arr = OSArray::withObjects((const OSObject **)&dict, 1, 2);
+               personalities->setObject(sym, arr);
+               arr->release();
+       }
 }
 
 /*********************************************************************
 * Initialize the IOCatalog object.
 *********************************************************************/
-bool IOCatalogue::init(OSArray * initArray)
+bool
+IOCatalogue::init(OSArray * initArray)
 {
-    OSDictionary         * dict;
-    OSObject * obj;
-
-    if ( !super::init() )
-        return false;
-
-    generation = 1;
-    
-    personalities = OSDictionary::withCapacity(32);
-    personalities->setOptions(OSCollection::kSort, OSCollection::kSort);
-    for (unsigned int idx = 0; (obj = initArray->getObject(idx)); idx++)
-    {
-       dict = OSDynamicCast(OSDictionary, obj);
-       if (!dict) continue;
-       OSKext::uniquePersonalityProperties(dict);
-        if( 0 == dict->getObject( gIOClassKey ))
-        {
-            IOLog("Missing or bad \"%s\" key\n",
-                    gIOClassKey->getCStringNoCopy());
-           continue;
-       }
-       dict->setObject("KernelConfigTable", kOSBooleanTrue);
-        addPersonality(dict);
-    }
-
-    gIOCatalogLock = IORWLockAlloc();
-    lock = gIOCatalogLock;
-
-    return true;
+       OSDictionary         * dict;
+       OSObject * obj;
+
+       if (!super::init()) {
+               return false;
+       }
+
+       generation = 1;
+
+       personalities = OSDictionary::withCapacity(32);
+       personalities->setOptions(OSCollection::kSort, OSCollection::kSort);
+       for (unsigned int idx = 0; (obj = initArray->getObject(idx)); idx++) {
+               dict = OSDynamicCast(OSDictionary, obj);
+               if (!dict) {
+                       continue;
+               }
+               OSKext::uniquePersonalityProperties(dict);
+               if (NULL == dict->getObject( gIOClassKey )) {
+                       IOLog("Missing or bad \"%s\" key\n",
+                           gIOClassKey->getCStringNoCopy());
+                       continue;
+               }
+               dict->setObject("KernelConfigTable", kOSBooleanTrue);
+               addPersonality(dict);
+       }
+
+       gIOCatalogLock = IORWLockAlloc();
+       lock = gIOCatalogLock;
+
+       return true;
 }
 
 /*********************************************************************
 * Release all resources used by IOCatalogue and deallocate.
 * This will probably never be called.
 *********************************************************************/
-void IOCatalogue::free( void )
+void
+IOCatalogue::free( void )
 {
-    panic("");
+       panic("");
 }
 
 /*********************************************************************
 *********************************************************************/
 OSOrderedSet *
 IOCatalogue::findDrivers(
-    IOService * service,
-    SInt32 * generationCount)
+       IOService * service,
+       SInt32 * generationCount)
 {
-    OSDictionary         * nextTable;
-    OSOrderedSet         * set;
-    OSArray              * array;
-    const OSMetaClass    * meta;
-    unsigned int           idx;
-
-    set = OSOrderedSet::withCapacity( 1, IOServiceOrdering,
-                                      (void *)gIOProbeScoreKey );
-    if( !set )
-       return( 0 );
+       OSDictionary         * nextTable;
+       OSOrderedSet         * set;
+       OSArray              * array;
+       const OSMetaClass    * meta;
+       unsigned int           idx;
+
+       set = OSOrderedSet::withCapacity( 1, IOServiceOrdering,
+           (void *)gIOProbeScoreKey );
+       if (!set) {
+               return NULL;
+       }
 
-    IORWLockRead(lock);
+       IORWLockRead(lock);
 
-    meta = service->getMetaClass();
-    while (meta)
-    {
-       array = (OSArray *) personalities->getObject(meta->getClassNameSymbol());
-       if (array) for (idx = 0; (nextTable = (OSDictionary *) array->getObject(idx)); idx++)
-       {
-            set->setObject(nextTable);
+       meta = service->getMetaClass();
+       while (meta) {
+               array = (OSArray *) personalities->getObject(meta->getClassNameSymbol());
+               if (array) {
+                       for (idx = 0; (nextTable = (OSDictionary *) array->getObject(idx)); idx++) {
+                               set->setObject(nextTable);
+                       }
+               }
+               if (meta == &IOService::gMetaClass) {
+                       break;
+               }
+               meta = meta->getSuperClass();
        }
-       if (meta == &IOService::gMetaClass) break;
-       meta = meta->getSuperClass();
-    }
 
-    *generationCount = getGenerationCount();
+       *generationCount = getGenerationCount();
 
-    IORWLockUnlock(lock);
+       IORWLockUnlock(lock);
 
-    return( set );
+       return set;
 }
 
 /*********************************************************************
@@ -233,46 +253,48 @@ IOCatalogue::findDrivers(
 *********************************************************************/
 OSOrderedSet *
 IOCatalogue::findDrivers(
-    OSDictionary * matching,
-    SInt32 * generationCount)
+       OSDictionary * matching,
+       SInt32 * generationCount)
 {
-    OSCollectionIterator * iter;
-    OSDictionary         * dict;
-    OSOrderedSet         * set;
-    OSArray              * array;
-    const OSSymbol       * key;
-    unsigned int           idx;
-
-    OSKext::uniquePersonalityProperties(matching);
-
-    set = OSOrderedSet::withCapacity( 1, IOServiceOrdering,
-                                      (void *)gIOProbeScoreKey );
-    if (!set) return (0);
-    iter = OSCollectionIterator::withCollection(personalities);
-    if (!iter) 
-    {
-       set->release();
-       return (0);
-    }
-
-    IORWLockRead(lock);
-    while ((key = (const OSSymbol *) iter->getNextObject()))
-    {
-        array = (OSArray *) personalities->getObject(key);
-        if (array) for (idx = 0; (dict = (OSDictionary *) array->getObject(idx)); idx++)
-        {
-          /* This comparison must be done with only the keys in the
-           * "matching" dict to enable general searches.
-           */
-           if ( dict->isEqualTo(matching, matching) )
-               set->setObject(dict);
-       }
-    }
-    *generationCount = getGenerationCount();
-    IORWLockUnlock(lock);
-
-    iter->release();
-    return set;
+       OSCollectionIterator * iter;
+       OSDictionary         * dict;
+       OSOrderedSet         * set;
+       OSArray              * array;
+       const OSSymbol       * key;
+       unsigned int           idx;
+
+       OSKext::uniquePersonalityProperties(matching);
+
+       set = OSOrderedSet::withCapacity( 1, IOServiceOrdering,
+           (void *)gIOProbeScoreKey );
+       if (!set) {
+               return NULL;
+       }
+       iter = OSCollectionIterator::withCollection(personalities);
+       if (!iter) {
+               set->release();
+               return NULL;
+       }
+
+       IORWLockRead(lock);
+       while ((key = (const OSSymbol *) iter->getNextObject())) {
+               array = (OSArray *) personalities->getObject(key);
+               if (array) {
+                       for (idx = 0; (dict = (OSDictionary *) array->getObject(idx)); idx++) {
+                               /* This comparison must be done with only the keys in the
+                                * "matching" dict to enable general searches.
+                                */
+                               if (dict->isEqualTo(matching, matching)) {
+                                       set->setObject(dict);
+                               }
+                       }
+               }
+       }
+       *generationCount = getGenerationCount();
+       IORWLockUnlock(lock);
+
+       iter->release();
+       return set;
 }
 
 /*********************************************************************
@@ -287,100 +309,104 @@ IOCatalogue::findDrivers(
 * xxx - during safe boot. That would be better implemented here.
 *********************************************************************/
 
-bool IOCatalogue::addDrivers(
-    OSArray * drivers,
-    bool doNubMatching)
+bool
+IOCatalogue::addDrivers(
+       OSArray * drivers,
+       bool doNubMatching)
 {
-    bool                   result = false;
-    OSCollectionIterator * iter = NULL;       // must release
-    OSOrderedSet         * set = NULL;        // must release
-    OSObject             * object = NULL;       // do not release
-    OSArray              * persons = NULL;    // do not release
-    
-    persons = OSDynamicCast(OSArray, drivers);
-    if (!persons) {
-        goto finish;
-    }
-    
-    set = OSOrderedSet::withCapacity( 10, IOServiceOrdering,
-        (void *)gIOProbeScoreKey );
-    if (!set) {
-        goto finish;
-    }
-
-    iter = OSCollectionIterator::withCollection(persons);
-    if (!iter) {
-        goto finish;
-    }
-
-   /* Start with success; clear it on an error.
-    */
-    result = true;
-
-    IORWLockWrite(lock);
-    while ( (object = iter->getNextObject()) ) {
-    
-        // xxx Deleted OSBundleModuleDemand check; will handle in other ways for SL
-
-        OSDictionary * personality = OSDynamicCast(OSDictionary, object);
-
-        SInt count;
-
-        if (!personality) {
-            IOLog("IOCatalogue::addDrivers() encountered non-dictionary; bailing.\n");
-            result = false;
-            break;
-        }
-
-        OSKext::uniquePersonalityProperties(personality);
-
-        // Add driver personality to catalogue.
-
-       OSArray * array = arrayForPersonality(personality);
-       if (!array) addPersonality(personality);
-       else
-       {       
-           count = array->getCount();
-           while (count--) {
-               OSDictionary * driver;
-               
-               // Be sure not to double up on personalities.
-               driver = (OSDictionary *)array->getObject(count);
-               
-              /* Unlike in other functions, this comparison must be exact!
-               * The catalogue must be able to contain personalities that
-               * are proper supersets of others.
-               * Do not compare just the properties present in one driver
-               * personality or the other.
-               */
-               if (personality->isEqualTo(driver)) {
-                   break;
+       bool                   result = false;
+       OSCollectionIterator * iter = NULL;   // must release
+       OSOrderedSet         * set = NULL;    // must release
+       OSObject             * object = NULL;   // do not release
+       OSArray              * persons = NULL;// do not release
+
+       persons = OSDynamicCast(OSArray, drivers);
+       if (!persons) {
+               goto finish;
+       }
+
+       set = OSOrderedSet::withCapacity( 10, IOServiceOrdering,
+           (void *)gIOProbeScoreKey );
+       if (!set) {
+               goto finish;
+       }
+
+       iter = OSCollectionIterator::withCollection(persons);
+       if (!iter) {
+               goto finish;
+       }
+
+       /* Start with success; clear it on an error.
+        */
+       result = true;
+
+       IORWLockWrite(lock);
+       while ((object = iter->getNextObject())) {
+               // xxx Deleted OSBundleModuleDemand check; will handle in other ways for SL
+
+               OSDictionary * personality = OSDynamicCast(OSDictionary, object);
+
+               SInt count;
+
+               if (!personality) {
+                       IOLog("IOCatalogue::addDrivers() encountered non-dictionary; bailing.\n");
+                       result = false;
+                       break;
                }
-           }
-           if (count >= 0) {
-               // its a dup
-               continue;
-           }
-           result = array->setObject(personality);
-           if (!result) {
-               break;
-           }
-        }
-
-       set->setObject(personality);        
-    }
-    // Start device matching.
-    if (result && doNubMatching && (set->getCount() > 0)) {
-        IOService::catalogNewDrivers(set);
-        generation++;
-    }
-    IORWLockUnlock(lock);
+
+               OSKext::uniquePersonalityProperties(personality);
+
+               // Add driver personality to catalogue.
+
+               OSArray * array = arrayForPersonality(personality);
+               if (!array) {
+                       addPersonality(personality);
+               } else {
+                       count = array->getCount();
+                       while (count--) {
+                               OSDictionary * driver;
+
+                               // Be sure not to double up on personalities.
+                               driver = (OSDictionary *)array->getObject(count);
+
+                               /* Unlike in other functions, this comparison must be exact!
+                                * The catalogue must be able to contain personalities that
+                                * are proper supersets of others.
+                                * Do not compare just the properties present in one driver
+                                * personality or the other.
+                                */
+                               if (personality->isEqualTo(driver)) {
+                                       break;
+                               }
+                       }
+                       if (count >= 0) {
+                               // its a dup
+                               continue;
+                       }
+                       result = array->setObject(personality);
+                       if (!result) {
+                               break;
+                       }
+               }
+
+               set->setObject(personality);
+       }
+       // Start device matching.
+       if (result && doNubMatching && (set->getCount() > 0)) {
+               IOService::catalogNewDrivers(set);
+               generation++;
+       }
+       IORWLockUnlock(lock);
 
 finish:
-    if (set)  set->release();
-    if (iter) iter->release();
+       if (set) {
+               set->release();
+       }
+       if (iter) {
+               iter->release();
+       }
 
-    return result;
+       return result;
 }
 
 /*********************************************************************
@@ -389,534 +415,648 @@ finish:
 *********************************************************************/
 bool
 IOCatalogue::removeDrivers(
-    OSDictionary * matching,
-    bool doNubMatching)
+       OSDictionary * matching,
+       bool doNubMatching)
 {
-    OSOrderedSet         * set;
-    OSCollectionIterator * iter;
-    OSDictionary         * dict;
-    OSArray              * array;
-    const OSSymbol       * key;
-    unsigned int           idx;
-
-    if ( !matching )
-        return false;
-    
-    set = OSOrderedSet::withCapacity(10,
-                                     IOServiceOrdering,
-                                     (void *)gIOProbeScoreKey);
-    if ( !set )
-        return false;
-    iter = OSCollectionIterator::withCollection(personalities);
-    if (!iter) 
-    {
-       set->release();
-       return (false);
-    }
-
-    IORWLockWrite(lock);
-    while ((key = (const OSSymbol *) iter->getNextObject()))
-    {
-        array = (OSArray *) personalities->getObject(key);
-        if (array) for (idx = 0; (dict = (OSDictionary *) array->getObject(idx)); idx++)
-        {
-           /* This comparison must be done with only the keys in the
-            * "matching" dict to enable general searches.
-            */
-            if ( dict->isEqualTo(matching, matching) ) {
-                set->setObject(dict);        
-                array->removeObject(idx);
-                idx--;
-            }
-        }
-        // Start device matching.
-        if ( doNubMatching && (set->getCount() > 0) ) {
-            IOService::catalogNewDrivers(set);
-            generation++;
-        }
-    }
-    IORWLockUnlock(lock);
-   
-    set->release();
-    iter->release();
-    
-    return true;
+       OSOrderedSet         * set;
+       OSCollectionIterator * iter;
+       OSDictionary         * dict;
+       OSArray              * array;
+       const OSSymbol       * key;
+       unsigned int           idx;
+
+       if (!matching) {
+               return false;
+       }
+
+       set = OSOrderedSet::withCapacity(10,
+           IOServiceOrdering,
+           (void *)gIOProbeScoreKey);
+       if (!set) {
+               return false;
+       }
+       iter = OSCollectionIterator::withCollection(personalities);
+       if (!iter) {
+               set->release();
+               return false;
+       }
+
+       IORWLockWrite(lock);
+       while ((key = (const OSSymbol *) iter->getNextObject())) {
+               array = (OSArray *) personalities->getObject(key);
+               if (array) {
+                       for (idx = 0; (dict = (OSDictionary *) array->getObject(idx)); idx++) {
+                               /* This comparison must be done with only the keys in the
+                                * "matching" dict to enable general searches.
+                                */
+                               if (dict->isEqualTo(matching, matching)) {
+                                       set->setObject(dict);
+                                       array->removeObject(idx);
+                                       idx--;
+                               }
+                       }
+               }
+               // Start device matching.
+               if (doNubMatching && (set->getCount() > 0)) {
+                       IOService::catalogNewDrivers(set);
+                       generation++;
+               }
+       }
+       IORWLockUnlock(lock);
+
+       set->release();
+       iter->release();
+
+       return true;
 }
 
 // Return the generation count.
-SInt32 IOCatalogue::getGenerationCount(void) const
+SInt32
+IOCatalogue::getGenerationCount(void) const
 {
-    return( generation );
+       return generation;
 }
 
-bool IOCatalogue::isModuleLoaded(OSString * moduleName) const
+// Check to see if kernel module has been loaded already, and request its load.
+bool
+IOCatalogue::isModuleLoaded(OSDictionary * driver, OSObject ** kextRef) const
 {
-    return isModuleLoaded(moduleName->getCStringNoCopy());
-}
+       OSString * moduleName = NULL;
+       OSString * publisherName = NULL;
+       OSReturn   ret;
 
-bool IOCatalogue::isModuleLoaded(const char * moduleName) const
-{
-    OSReturn ret;
-    ret = OSKext::loadKextWithIdentifier(moduleName);
-    if (kOSKextReturnDeferred == ret) {
-        // a request has been queued but the module isn't necessarily 
-        // loaded yet, so stall.
-        return false;
-    }
-    // module is present or never will be 
-    return true;
-}
+       if (kextRef) {
+               *kextRef = NULL;
+       }
+       if (!driver) {
+               return false;
+       }
 
-// Check to see if module has been loaded already.
-bool IOCatalogue::isModuleLoaded(OSDictionary * driver) const
-{
-    OSString             * moduleName = NULL;
-    OSString             * publisherName = NULL;
-
-    if ( !driver )
-        return false;
-
-    /* The personalities of codeless kexts often contain the bundle ID of the
-     * kext they reference, and not the bundle ID of the codeless kext itself.
-     * The prelinked kernel needs to know the bundle ID of the codeless kext
-     * so it can include these personalities, so OSKext stores that bundle ID
-     * in the IOPersonalityPublisher key, and we record it as requested here.
-     */
-    publisherName = OSDynamicCast(OSString, 
-        driver->getObject(kIOPersonalityPublisherKey));
-    OSKext::recordIdentifierRequest(publisherName);
-
-    moduleName = OSDynamicCast(OSString, driver->getObject(gIOModuleIdentifierKey));
-    if ( moduleName )
-        return isModuleLoaded(moduleName);
-
-   /* If a personality doesn't hold the "CFBundleIdentifier" key
-    * it is assumed to be an "in-kernel" driver.
-    */
-    return true;
+       /* The personalities of codeless kexts often contain the bundle ID of the
+        * kext they reference, and not the bundle ID of the codeless kext itself.
+        * The prelinked kernel needs to know the bundle ID of the codeless kext
+        * so it can include these personalities, so OSKext stores that bundle ID
+        * in the IOPersonalityPublisher key, and we record it as requested here.
+        */
+       publisherName = OSDynamicCast(OSString,
+           driver->getObject(kIOPersonalityPublisherKey));
+       OSKext::recordIdentifierRequest(publisherName);
+
+       moduleName = OSDynamicCast(OSString, driver->getObject(gIOModuleIdentifierKernelKey));
+       if (moduleName) {
+               ret = OSKext::loadKextWithIdentifier(moduleName, kextRef);
+               if (kOSKextReturnDeferred == ret) {
+                       // a request has been queued but the module isn't necessarily
+                       // loaded yet, so stall.
+                       return false;
+               }
+               OSString *moduleDextName = OSDynamicCast(OSString, driver->getObject(gIOModuleIdentifierKey));
+               if (moduleDextName && !(moduleName->isEqualTo(moduleDextName))) {
+                       OSObject *dextRef = NULL;
+                       ret = OSKext::loadKextWithIdentifier(moduleDextName, &dextRef);
+                       OSSafeReleaseNULL(dextRef);
+               }
+               // module is present or never will be
+               return true;
+       }
+
+       /* If a personality doesn't hold the "CFBundleIdentifier" or "CFBundleIdentifierKernel" key
+        * it is assumed to be an "in-kernel" driver.
+        */
+       return true;
 }
 
 /* This function is called after a module has been loaded.
  * Is invoked from user client call, ultimately from IOKitLib's
  * IOCatalogueModuleLoaded(). Sent from kextd.
  */
-void IOCatalogue::moduleHasLoaded(OSString * moduleName)
+void
+IOCatalogue::moduleHasLoaded(const OSSymbol * moduleName)
 {
-    OSDictionary * dict;
-
-    dict = OSDictionary::withCapacity(2);
-    dict->setObject(gIOModuleIdentifierKey, moduleName);
-    startMatching(dict);
-    dict->release();
+       startMatching(moduleName);
 
-    (void) OSKext::setDeferredLoadSucceeded();
-    (void) OSKext::considerRebuildOfPrelinkedKernel();
+       (void) OSKext::setDeferredLoadSucceeded();
+       (void) OSKext::considerRebuildOfPrelinkedKernel();
 }
 
-void IOCatalogue::moduleHasLoaded(const char * moduleName)
+void
+IOCatalogue::moduleHasLoaded(const char * moduleName)
 {
-    OSString * name;
+       const OSSymbol * name;
 
-    name = OSString::withCString(moduleName);
-    moduleHasLoaded(name);
-    name->release();
+       name = OSSymbol::withCString(moduleName);
+       moduleHasLoaded(name);
+       name->release();
 }
 
 // xxx - return is really OSReturn/kern_return_t
-IOReturn IOCatalogue::unloadModule(OSString * moduleName) const
+IOReturn
+IOCatalogue::unloadModule(OSString * moduleName) const
 {
-    return OSKext::removeKextWithIdentifier(moduleName->getCStringNoCopy());
+       return OSKext::removeKextWithIdentifier(moduleName->getCStringNoCopy());
 }
 
-IOReturn IOCatalogue::_terminateDrivers(OSDictionary * matching)
+IOReturn
+IOCatalogue::_terminateDrivers(OSDictionary * matching)
 {
-    OSDictionary         * dict;
-    OSIterator           * iter;
-    IOService            * service;
-    IOReturn               ret;
-
-    if ( !matching )
-        return kIOReturnBadArgument;
-
-    ret = kIOReturnSuccess;
-    dict = 0;
-    iter = IORegistryIterator::iterateOver(gIOServicePlane,
-                                kIORegistryIterateRecursively);
-    if ( !iter )
-        return kIOReturnNoMemory;
-
-    OSKext::uniquePersonalityProperties( matching );
-
-    // terminate instances.
-    do {
-        iter->reset();
-        while( (service = (IOService *)iter->getNextObject()) ) {
-            dict = service->getPropertyTable();
-            if ( !dict )
-                continue;
-
-           /* Terminate only for personalities that match the matching dictionary.
-            * This comparison must be done with only the keys in the
-            * "matching" dict to enable general matching.
-            */
-            if ( !dict->isEqualTo(matching, matching) )
-                 continue;
-
-            if ( !service->terminate(kIOServiceRequired|kIOServiceSynchronous) ) {
-                ret = kIOReturnUnsupported;
-                break;
-            }
-        }
-    } while( !service && !iter->isValid());
-    iter->release();
-
-    return ret;
+       OSDictionary         * dict;
+       OSIterator           * iter;
+       IOService            * service;
+       IOReturn               ret;
+
+       if (!matching) {
+               return kIOReturnBadArgument;
+       }
+
+       ret = kIOReturnSuccess;
+       dict = NULL;
+       iter = IORegistryIterator::iterateOver(gIOServicePlane,
+           kIORegistryIterateRecursively);
+       if (!iter) {
+               return kIOReturnNoMemory;
+       }
+
+       OSKext::uniquePersonalityProperties( matching );
+
+       // terminate instances.
+       do {
+               iter->reset();
+               while ((service = (IOService *)iter->getNextObject())) {
+                       dict = service->getPropertyTable();
+                       if (!dict) {
+                               continue;
+                       }
+
+                       /* Terminate only for personalities that match the matching dictionary.
+                        * This comparison must be done with only the keys in the
+                        * "matching" dict to enable general matching.
+                        */
+                       if (!dict->isEqualTo(matching, matching)) {
+                               continue;
+                       }
+
+                       if (!service->terminate(kIOServiceRequired | kIOServiceSynchronous)) {
+                               ret = kIOReturnUnsupported;
+                               break;
+                       }
+               }
+       } while (!service && !iter->isValid());
+       iter->release();
+
+       return ret;
 }
 
-IOReturn IOCatalogue::_removeDrivers(OSDictionary * matching)
+IOReturn
+IOCatalogue::_removeDrivers(OSDictionary * matching)
 {
-    IOReturn               ret = kIOReturnSuccess;
-    OSCollectionIterator * iter;
-    OSDictionary         * dict;
-    OSArray              * array;
-    const OSSymbol       * key;
-    unsigned int           idx;
-
-    // remove configs from catalog.
-
-    iter = OSCollectionIterator::withCollection(personalities);
-    if (!iter) return (kIOReturnNoMemory);
-
-    while ((key = (const OSSymbol *) iter->getNextObject()))
-    {
-        array = (OSArray *) personalities->getObject(key);
-        if (array) for (idx = 0; (dict = (OSDictionary *) array->getObject(idx)); idx++)
-        {
-
-           /* Remove from the catalogue's array any personalities
-            * that match the matching dictionary.
-            * This comparison must be done with only the keys in the
-            * "matching" dict to enable general matching.
-            */
-            if (dict->isEqualTo(matching, matching))
-            {
-                array->removeObject(idx);
-                idx--;
-            }
-        }
-    }
-    iter->release();
-
-    return ret;
+       IOReturn               ret = kIOReturnSuccess;
+       OSCollectionIterator * iter;
+       OSDictionary         * dict;
+       OSArray              * array;
+       const OSSymbol       * key;
+       unsigned int           idx;
+
+       // remove configs from catalog.
+
+       iter = OSCollectionIterator::withCollection(personalities);
+       if (!iter) {
+               return kIOReturnNoMemory;
+       }
+
+       while ((key = (const OSSymbol *) iter->getNextObject())) {
+               array = (OSArray *) personalities->getObject(key);
+               if (array) {
+                       for (idx = 0; (dict = (OSDictionary *) array->getObject(idx)); idx++) {
+                               /* Remove from the catalogue's array any personalities
+                                * that match the matching dictionary.
+                                * This comparison must be done with only the keys in the
+                                * "matching" dict to enable general matching.
+                                */
+                               if (dict->isEqualTo(matching, matching)) {
+                                       array->removeObject(idx);
+                                       idx--;
+                               }
+                       }
+               }
+       }
+       iter->release();
+
+       return ret;
 }
 
-IOReturn IOCatalogue::terminateDrivers(OSDictionary * matching)
+IOReturn
+IOCatalogue::terminateDrivers(OSDictionary * matching)
 {
-    IOReturn ret;
+       IOReturn ret;
 
-    ret = _terminateDrivers(matching);
-    IORWLockWrite(lock);
-    if (kIOReturnSuccess == ret)
-       ret = _removeDrivers(matching);
-    IORWLockUnlock(lock);
+       ret = _terminateDrivers(matching);
+       IORWLockWrite(lock);
+       if (kIOReturnSuccess == ret) {
+               ret = _removeDrivers(matching);
+       }
+       IORWLockUnlock(lock);
 
-    return ret;
+       return ret;
 }
 
-IOReturn IOCatalogue::terminateDriversForModule(
-    OSString * moduleName,
-    bool unload)
+IOReturn
+IOCatalogue::terminateDriversForModule(
+       OSString * moduleName,
+       bool unload)
 {
-    IOReturn ret;
-    OSDictionary * dict;
-    bool isLoaded = false;
-
-   /* Check first if the kext currently has any linkage dependents;
-    * in such a case the unload would fail so let's not terminate any
-    * IOServices (since doing so typically results in a panic when there
-    * are loaded dependencies). Note that we aren't locking the kext here
-    * so it might lose or gain dependents by the time we call unloadModule();
-    * I think that's ok, our unload can fail if a kext comes in on top of
-    * this one even after we've torn down IOService objects. Conversely,
-    * if we fail the unload here and then lose a library, the autounload
-    * thread will get us in short order.
-    */
-    if (OSKext::isKextWithIdentifierLoaded(moduleName->getCStringNoCopy())) {
-    
-        isLoaded = true;
-
-        if (!OSKext::canUnloadKextWithIdentifier(moduleName,
-            /* checkClasses */ false)) {
-            ret = kOSKextReturnInUse;
-            goto finish;
-        }
-    }
-    dict = OSDictionary::withCapacity(1);
-    if (!dict) {
-        ret = kIOReturnNoMemory;
-        goto finish;
-    }
-
-    dict->setObject(gIOModuleIdentifierKey, moduleName);
-
-    ret = _terminateDrivers(dict);
-    
-   /* No goto between IOLock calls!
-    */
-    IORWLockWrite(lock);
-    if (kIOReturnSuccess == ret) {
-        ret = _removeDrivers(dict);
-    }
-
-    // Unload the module itself.
-    if (unload && isLoaded && ret == kIOReturnSuccess) {
-        ret = unloadModule(moduleName);
-    }
-
-    IORWLockUnlock(lock);
-
-    dict->release();
+       IOReturn ret;
+       OSDictionary * dict;
+       bool isLoaded = false;
+
+       /* Check first if the kext currently has any linkage dependents;
+        * in such a case the unload would fail so let's not terminate any
+        * IOServices (since doing so typically results in a panic when there
+        * are loaded dependencies). Note that we aren't locking the kext here
+        * so it might lose or gain dependents by the time we call unloadModule();
+        * I think that's ok, our unload can fail if a kext comes in on top of
+        * this one even after we've torn down IOService objects. Conversely,
+        * if we fail the unload here and then lose a library, the autounload
+        * thread will get us in short order.
+        */
+       if (OSKext::isKextWithIdentifierLoaded(moduleName->getCStringNoCopy())) {
+               isLoaded = true;
+
+               if (!OSKext::canUnloadKextWithIdentifier(moduleName,
+                   /* checkClasses */ false)) {
+                       ret = kOSKextReturnInUse;
+                       goto finish;
+               }
+       }
+       dict = OSDictionary::withCapacity(1);
+       if (!dict) {
+               ret = kIOReturnNoMemory;
+               goto finish;
+       }
+
+       dict->setObject(gIOModuleIdentifierKey, moduleName);
+
+       ret = _terminateDrivers(dict);
+
+       /* No goto between IOLock calls!
+        */
+       IORWLockWrite(lock);
+       if (kIOReturnSuccess == ret) {
+               ret = _removeDrivers(dict);
+       }
+
+       // Unload the module itself.
+       if (unload && isLoaded && ret == kIOReturnSuccess) {
+               ret = unloadModule(moduleName);
+       }
+
+       IORWLockUnlock(lock);
+
+       dict->release();
 
 finish:
-    return ret;
+       return ret;
+}
+
+IOReturn
+IOCatalogue::terminateDriversForModule(
+       const char * moduleName,
+       bool unload)
+{
+       OSString * name;
+       IOReturn ret;
+
+       name = OSString::withCString(moduleName);
+       if (!name) {
+               return kIOReturnNoMemory;
+       }
+
+       ret = terminateDriversForModule(name, unload);
+       name->release();
+
+       return ret;
 }
 
-IOReturn IOCatalogue::terminateDriversForModule(
-    const char * moduleName,
-    bool unload)
+#if defined(__i386__) || defined(__x86_64__)
+bool
+IOCatalogue::startMatching( OSDictionary * matching )
 {
-    OSString * name;
-    IOReturn ret;
+       OSOrderedSet         * set;
+
+       if (!matching) {
+               return false;
+       }
+
+       set = OSOrderedSet::withCapacity(10, IOServiceOrdering,
+           (void *)gIOProbeScoreKey);
+       if (!set) {
+               return false;
+       }
 
-    name = OSString::withCString(moduleName);
-    if ( !name )
-        return kIOReturnNoMemory;
+       IORWLockRead(lock);
+
+       personalities->iterateObjects(^bool (const OSSymbol * key, OSObject * value) {
+               OSArray      * array;
+               OSDictionary * dict;
+               unsigned int   idx;
+
+               array = (OSArray *) value;
+               for (idx = 0; (dict = (OSDictionary *) array->getObject(idx)); idx++) {
+                       /* This comparison must be done with only the keys in the
+                        * "matching" dict to enable general matching.
+                        */
+                       if (dict->isEqualTo(matching, matching)) {
+                               set->setObject(dict);
+                       }
+               }
+               return false;
+       });
+
+       // Start device matching.
+       if (set->getCount() > 0) {
+               IOService::catalogNewDrivers(set);
+               generation++;
+       }
 
-    ret = terminateDriversForModule(name, unload);
-    name->release();
+       IORWLockUnlock(lock);
 
-    return ret;
+       set->release();
+
+       return true;
 }
+#endif /* defined(__i386__) || defined(__x86_64__) */
 
-bool IOCatalogue::startMatching( OSDictionary * matching )
+bool
+IOCatalogue::startMatching( const OSSymbol * moduleName )
 {
-    OSCollectionIterator * iter;
-    OSDictionary         * dict;
-    OSOrderedSet         * set;
-    OSArray              * array;
-    const OSSymbol *       key;
-    unsigned int           idx;
-    
-    if ( !matching )
-        return false;
-
-    set = OSOrderedSet::withCapacity(10, IOServiceOrdering,
-                                     (void *)gIOProbeScoreKey);
-    if ( !set )
-        return false;
-
-    iter = OSCollectionIterator::withCollection(personalities);
-    if (!iter) 
-    {
-       set->release();
-        return false;
-    }
-
-    IORWLockRead(lock);
-
-    while ((key = (const OSSymbol *) iter->getNextObject()))
-    {
-        array = (OSArray *) personalities->getObject(key);
-        if (array) for (idx = 0; (dict = (OSDictionary *) array->getObject(idx)); idx++)
-        {
-          /* This comparison must be done with only the keys in the
-           * "matching" dict to enable general matching.
-           */
-            if (dict->isEqualTo(matching, matching)) {
-                set->setObject(dict);
-            }        
-        }
-    }
-
-    // Start device matching.
-    if ( set->getCount() > 0 ) {
-        IOService::catalogNewDrivers(set);
-        generation++;
-    }
-
-    IORWLockUnlock(lock);
-
-    set->release();
-    iter->release();
-
-    return true;
+       OSOrderedSet         * set;
+
+       if (!moduleName) {
+               return false;
+       }
+
+       set = OSOrderedSet::withCapacity(10, IOServiceOrdering,
+           (void *)gIOProbeScoreKey);
+       if (!set) {
+               return false;
+       }
+
+       IORWLockRead(lock);
+
+       personalities->iterateObjects(^bool (const OSSymbol * key, OSObject * value) {
+               OSArray      * array;
+               OSDictionary * dict;
+               OSObject     * obj;
+               unsigned int   idx;
+
+               array = (OSArray *) value;
+               for (idx = 0; (dict = (OSDictionary *) array->getObject(idx)); idx++) {
+                       obj = dict->getObject(gIOModuleIdentifierKernelKey);
+                       if (obj && moduleName->isEqualTo(obj)) {
+                               set->setObject(dict);
+                       }
+               }
+               return false;
+       });
+
+       // Start device matching.
+       if (set->getCount() > 0) {
+               IOService::catalogNewDrivers(set);
+               generation++;
+       }
+
+       IORWLockUnlock(lock);
+
+       set->release();
+
+       return true;
 }
 
-void IOCatalogue::reset(void)
+void
+IOCatalogue::reset(void)
 {
-    IOCatalogue::resetAndAddDrivers(/* no drivers; true reset */ NULL,
-        /* doMatching */ false);
-    return;
+       IOCatalogue::resetAndAddDrivers(/* no drivers; true reset */ NULL,
+           /* doMatching */ false);
+       return;
 }
 
-bool IOCatalogue::resetAndAddDrivers(OSArray * drivers, bool doNubMatching)
+bool
+IOCatalogue::resetAndAddDrivers(OSArray * drivers, bool doNubMatching)
 {
-    bool                   result              = false;
-    OSArray              * newPersonalities    = NULL;  // do not release
-    OSCollectionIterator * iter                = NULL;  // must release
-    OSOrderedSet         * matchSet            = NULL;  // must release
-    const OSSymbol       * key;
-    OSArray              * array;
-    OSDictionary         * thisNewPersonality  = NULL;  // do not release
-    OSDictionary         * thisOldPersonality  = NULL;  // do not release
-    signed int             idx, newIdx;
-
-    if (drivers) {
-        newPersonalities = OSDynamicCast(OSArray, drivers);
-        if (!newPersonalities) {
-            goto finish;
-        }
-        
-        matchSet = OSOrderedSet::withCapacity(10, IOServiceOrdering,
-            (void *)gIOProbeScoreKey);
-        if (!matchSet) {
-            goto finish;
-        }
-        iter = OSCollectionIterator::withCollection(personalities);
-        if (!iter) {
-            goto finish;
-        }
-    }
-
-    result = true;
-
-    IOLog("Resetting IOCatalogue.\n");
-
-   /* No goto finish from here to unlock.
-    */
-    IORWLockWrite(lock);
-    
-    while ((key = (const OSSymbol *) iter->getNextObject()))
-    {
-        array = (OSArray *) personalities->getObject(key);
-        if (!array) continue;
-        for (idx = 0; (thisOldPersonality = (OSDictionary *) array->getObject(idx)); idx++)
-        {
-            if (thisOldPersonality->getObject("KernelConfigTable")) continue;
-            if (newPersonalities)
-            for  (newIdx = 0;
-                 (thisNewPersonality = (OSDictionary *) newPersonalities->getObject(newIdx));
-                 newIdx++)
-            {
-              /* Unlike in other functions, this comparison must be exact!
-            * The catalogue must be able to contain personalities that
-            * are proper supersets of others.
-            * Do not compare just the properties present in one driver
-            * personality or the other.
-            */
-                if (OSDynamicCast(OSDictionary, thisNewPersonality) == NULL) {
-                    /* skip thisNewPersonality if it is not an OSDictionary */
-                    continue;
-                }
-                if (thisNewPersonality->isEqualTo(thisOldPersonality))
-                    break;
-            }
-            if (thisNewPersonality)
-            {
-                // dup, ignore
-                newPersonalities->removeObject(newIdx);
-            }
-            else
-            {
-                // not in new set - remove
-                // only remove dictionary if this module in not loaded - 9953845
-                if ( isModuleLoaded(thisOldPersonality) == false ) 
-                {
-                    if (matchSet)  matchSet->setObject(thisOldPersonality);
-                    array->removeObject(idx);
-                    idx--;
-                }
-            }
-        }
-    }
-
-    // add new
-    for (newIdx = 0;
-         (thisNewPersonality = (OSDictionary *) newPersonalities->getObject(newIdx));
-         newIdx++)
-    {
-         if (OSDynamicCast(OSDictionary, thisNewPersonality) == NULL) {
-             /* skip thisNewPersonality if it is not an OSDictionary */
-             continue;
-         }
-
-         OSKext::uniquePersonalityProperties(thisNewPersonality);
-         addPersonality(thisNewPersonality);
-         matchSet->setObject(thisNewPersonality);
-    }
-
-   /* Finally, start device matching on all new & removed personalities.
-    */
-    if (result && doNubMatching && (matchSet->getCount() > 0)) {
-        IOService::catalogNewDrivers(matchSet);
-        generation++;
-    }
-
-    IORWLockUnlock(lock);
+       bool                   result              = false;
+       OSArray              * newPersonalities    = NULL;// do not release
+       OSCollectionIterator * iter                = NULL;// must release
+       OSOrderedSet         * matchSet            = NULL;// must release
+       const OSSymbol       * key;
+       OSArray              * array;
+       OSDictionary         * thisNewPersonality   = NULL;// do not release
+       OSDictionary         * thisOldPersonality   = NULL;// do not release
+       OSDictionary         * myKexts              = NULL;// must release
+       signed int             idx, newIdx;
+
+       if (drivers) {
+               newPersonalities = OSDynamicCast(OSArray, drivers);
+               if (!newPersonalities) {
+                       goto finish;
+               }
+       }
+       matchSet = OSOrderedSet::withCapacity(10, IOServiceOrdering,
+           (void *)gIOProbeScoreKey);
+       if (!matchSet) {
+               goto finish;
+       }
+       iter = OSCollectionIterator::withCollection(personalities);
+       if (!iter) {
+               goto finish;
+       }
+
+       /* need copy of loaded kexts so we can check if for loaded modules without
+        * taking the OSKext lock.  There is a potential of deadlocking if we get
+        * an OSKext via the normal path.  See 14672140.
+        */
+       myKexts = OSKext::copyKexts();
+
+       result = true;
+
+       IOLog("Resetting IOCatalogue.\n");
+
+       /* No goto finish from here to unlock.
+        */
+       IORWLockWrite(lock);
+
+       while ((key = (const OSSymbol *) iter->getNextObject())) {
+               array = (OSArray *) personalities->getObject(key);
+               if (!array) {
+                       continue;
+               }
+
+               for (idx = 0;
+                   (thisOldPersonality = (OSDictionary *) array->getObject(idx));
+                   idx++) {
+                       if (thisOldPersonality->getObject("KernelConfigTable")) {
+                               continue;
+                       }
+                       thisNewPersonality = NULL;
+
+                       if (newPersonalities) {
+                               for (newIdx = 0;
+                                   (thisNewPersonality = (OSDictionary *) newPersonalities->getObject(newIdx));
+                                   newIdx++) {
+                                       /* Unlike in other functions, this comparison must be exact!
+                                        * The catalogue must be able to contain personalities that
+                                        * are proper supersets of others.
+                                        * Do not compare just the properties present in one driver
+                                        * personality or the other.
+                                        */
+                                       if (OSDynamicCast(OSDictionary, thisNewPersonality) == NULL) {
+                                               /* skip thisNewPersonality if it is not an OSDictionary */
+                                               continue;
+                                       }
+                                       if (thisNewPersonality->isEqualTo(thisOldPersonality)) {
+                                               break;
+                                       }
+                               }
+                       }
+                       if (thisNewPersonality) {
+                               // dup, ignore
+                               newPersonalities->removeObject(newIdx);
+                       } else {
+                               // not in new set - remove
+                               // only remove dictionary if this module in not loaded - 9953845
+                               if (isModuleLoadedNoOSKextLock(myKexts, thisOldPersonality) == false) {
+                                       if (matchSet) {
+                                               matchSet->setObject(thisOldPersonality);
+                                       }
+                                       array->removeObject(idx);
+                                       idx--;
+                               }
+                       }
+               } // for...
+       } // while...
+
+       // add new
+       if (newPersonalities) {
+               for (newIdx = 0;
+                   (thisNewPersonality = (OSDictionary *) newPersonalities->getObject(newIdx));
+                   newIdx++) {
+                       if (OSDynamicCast(OSDictionary, thisNewPersonality) == NULL) {
+                               /* skip thisNewPersonality if it is not an OSDictionary */
+                               continue;
+                       }
+
+                       OSKext::uniquePersonalityProperties(thisNewPersonality);
+                       addPersonality(thisNewPersonality);
+                       matchSet->setObject(thisNewPersonality);
+               }
+       }
+
+       /* Finally, start device matching on all new & removed personalities.
+        */
+       if (result && doNubMatching && (matchSet->getCount() > 0)) {
+               IOService::catalogNewDrivers(matchSet);
+               generation++;
+       }
+
+       IORWLockUnlock(lock);
 
 finish:
-    if (matchSet) matchSet->release();
-    if (iter)     iter->release();
+       if (matchSet) {
+               matchSet->release();
+       }
+       if (iter) {
+               iter->release();
+       }
+       if (myKexts) {
+               myKexts->release();
+       }
 
-    return result;
+       return result;
 }
 
-bool IOCatalogue::serialize(OSSerialize * s) const
+bool
+IOCatalogue::serialize(OSSerialize * s) const
 {
-    if ( !s )
-        return false;
+       if (!s) {
+               return false;
+       }
 
-    return super::serialize(s);
+       return super::serialize(s);
 }
 
-bool IOCatalogue::serializeData(IOOptionBits kind, OSSerialize * s) const
+bool
+IOCatalogue::serializeData(IOOptionBits kind, OSSerialize * s) const
 {
-    kern_return_t kr = kIOReturnSuccess;
+       kern_return_t kr = kIOReturnSuccess;
 
-    switch ( kind )
-    {
-        case kIOCatalogGetContents:
-            kr = KERN_NOT_SUPPORTED;
-            break;
+       switch (kind) {
+       case kIOCatalogGetContents:
+               kr = KERN_NOT_SUPPORTED;
+               break;
 
-        case kIOCatalogGetModuleDemandList:
-            kr = KERN_NOT_SUPPORTED;
-            break;
+       case kIOCatalogGetModuleDemandList:
+               kr = KERN_NOT_SUPPORTED;
+               break;
 
-        case kIOCatalogGetCacheMissList:
-            kr = KERN_NOT_SUPPORTED;
-            break;
+       case kIOCatalogGetCacheMissList:
+               kr = KERN_NOT_SUPPORTED;
+               break;
 
-        case kIOCatalogGetROMMkextList:
-            kr = KERN_NOT_SUPPORTED;
-            break;
+       case kIOCatalogGetROMMkextList:
+               kr = KERN_NOT_SUPPORTED;
+               break;
 
-        default:
-            kr = kIOReturnBadArgument;
-            break;
-    }
+       default:
+               kr = kIOReturnBadArgument;
+               break;
+       }
 
-    return kr;
+       return kr;
 }
 
+/* isModuleLoadedNoOSKextLock - used to check to see if a kext is loaded
+ * without taking the OSKext lock.  We use this to avoid the problem
+ * where taking the IOCatalog lock then the OSKext lock will dealock when
+ * a kext load or unload is happening at the same time as IOCatalog changing.
+ *
+ * theKexts - is a dictionary of current kexts (from OSKext::copyKexts) with
+ *      key set to the kext bundle ID and value set to an OSKext object
+ * theModuleDict - is an IOKit personality dictionary for a given module (kext)
+ */
+static bool
+isModuleLoadedNoOSKextLock(OSDictionary *theKexts,
+    OSDictionary *theModuleDict)
+{
+       bool                    myResult = false;
+       const OSString *        myBundleID = NULL;// do not release
+       OSKext *                myKext = NULL;  // do not release
+
+       if (theKexts == NULL || theModuleDict == NULL) {
+               return myResult;
+       }
+
+       // gIOModuleIdentifierKey is "CFBundleIdentifier"
+       myBundleID = OSDynamicCast(OSString,
+           theModuleDict->getObject(gIOModuleIdentifierKey));
+       if (myBundleID == NULL) {
+               return myResult;
+       }
+
+       myKext = OSDynamicCast(OSKext, theKexts->getObject(myBundleID->getCStringNoCopy()));
+       if (myKext) {
+               myResult = myKext->isLoaded();
+       }
+
+       return myResult;
+}
+
+
 #if PRAGMA_MARK
 #pragma mark Obsolete Kext Loading Stuff
 #endif
 /*********************************************************************
-**********************************************************************
-***                  BINARY COMPATIBILITY SECTION                  ***
-**********************************************************************
-**********************************************************************
-* These functions are no longer used are necessary for C++ binary
-* compatibility on i386.
-**********************************************************************/
+ **********************************************************************
+ ***                  BINARY COMPATIBILITY SECTION                  ***
+ **********************************************************************
+ **********************************************************************
+ * These functions are no longer used are necessary for C++ binary
+ * compatibility on i386.
+ **********************************************************************/