Search code examples
ioscore-datansmanagedobjectmodel

Crash when reusing NSFetchRequest across multiple contexts


Edit: The original title for this thread was "Crash when creating a second managed object model", but the problem was solved and the problem turned out to be as described in the title above.

TL;DR: Do not reuse NSFetchRequests across multiple NSManagedObjectContext objects. See answer.


We are seeing a problem that seems to be related to NSEntityDescription reuse gone bad.

We have an app that uses Core Data per user, and allows users to login and logout. Each session has something we call Core Data stack, which loads a managed object model, store coordinator and the context hierarchy we use. When the user logs out, we release the stack, and the model, coordinator and contexts are released. However, when the users logs in again, we get a crash. Enabling zombies, we see the following:

2014-04-30 11:31:50.755 CPiPhoneSSLVPN[65981:907] *** -[NSEntityDescription name]: message sent to deallocated instance 0x19888170

This happens whenever we attempt to perform a fetch. I played around with the Core Data stack, and made the model static (it is loaded on demand once, and is never released) and this works as expected. Problem is, our app modifies at runtime the model entities to conform to a security policy (for example, we make certain properties transient so that they are not stored to disk). So making the model is not a viable solution.

Here is the backtrace of the crash:

* thread #1: tid = 0x3c6736, 0x02793411 CoreFoundation`___forwarding___ + 769, name = 'Main', queue = 'com.apple.main-thread', stop reason = EXC_BREAKPOINT (code=EXC_I386_BPT, subcode=0x0)
  * frame #0: 0x02793411 CoreFoundation`___forwarding___ + 769
    frame #1: 0x027930ee CoreFoundation`__forwarding_prep_0___ + 14
    frame #2: 0x0319e71e CoreData`fetchPlanAllocateInitialize + 206
    frame #3: 0x0319e2e4 CoreData`-[NSSQLCore objectsForFetchRequest:inContext:] + 644
    frame #4: 0x0319ddcf CoreData`-[NSSQLCore executeRequest:withContext:error:] + 383
    frame #5: 0x0319d7f2 CoreData`-[NSPersistentStoreCoordinator executeRequest:withContext:error:] + 4466
    frame #6: 0x0319af56 CoreData`-[NSManagedObjectContext executeFetchRequest:error:] + 566
    frame #7: 0x031efd86 CoreData`-[NSManagedObjectContext(_NestedContextSupport) _parentObjectsForFetchRequest:inContext:error:] + 502
    frame #8: 0x0326da14 CoreData`__82-[NSManagedObjectContext(_NestedContextSupport) executeRequest:withContext:error:]_block_invoke + 676
    frame #9: 0x031efb81 CoreData`internalBlockToNSManagedObjectContextPerform + 17
    frame #10: 0x06c2d4d0 libdispatch.dylib`_dispatch_client_callout + 14
    frame #11: 0x06c1c439 libdispatch.dylib`_dispatch_barrier_sync_f_slow_invoke + 80
    frame #12: 0x06c2d4d0 libdispatch.dylib`_dispatch_client_callout + 14
    frame #13: 0x06c1b726 libdispatch.dylib`_dispatch_main_queue_callback_4CF + 340
    frame #14: 0x0280843e CoreFoundation`__CFRUNLOOP_IS_SERVICING_THE_MAIN_DISPATCH_QUEUE__ + 14
    frame #15: 0x027495cb CoreFoundation`__CFRunLoopRun + 1963
    frame #16: 0x027489d3 CoreFoundation`CFRunLoopRunSpecific + 467
    frame #17: 0x027487eb CoreFoundation`CFRunLoopRunInMode + 123
    frame #18: 0x062335ee GraphicsServices`GSEventRunModal + 192
    frame #19: 0x0623342b GraphicsServices`GSEventRun + 104
    frame #20: 0x048ddf9b UIKit`UIApplicationMain + 1225
    frame #21: 0x0003b0e6 CPiPhoneSSLVPN`main(argc=1, argv=0xbfffed7c) + 86 at main.m:17

Has anyone seen this kind of problem? Any tips?

I've opened bug report 16767152 for this issue.


Solution

  • The problem was we were using a cached NSFetchRequest. It seems the fetch requests caches the entity internally on first use by a context, and later if reused, it naively attempts to use its cache. This created the above error.