So I'm trying to convert an old project to Automatic Reference Counting. I'm trying to use the conversion tool that xCode has but it says to fix a couple things before it can convert. I have no idea how to fix this error. It's in the implementation of the keychain file. This method is the one that returns the error, specifically the line with the SecItemCopyMatching. The error I am getting says: " Cast of an indirect pointer to an Objective-C pointer to 'CFTypeRef*' (aka 'const void**') is disallowed with ARC. I've been looking all over google, apple docs, and a bunch of other crap and can't find a better way to fetch an existing data dictionary in the keychain. Any help appreciated. Thanks!
-(NSMutableDictionary*)fetchDictionary {
NSMutableDictionary *genericPasswordQuery = [self buildSearchQuery];
NSMutableDictionary *outDictionary = nil;
OSStatus status = SecItemCopyMatching((__bridge_retained CFDictionaryRef)genericPasswordQuery, (CFTypeRef*)&outDictionary);
if (DEBUG) printf("FETCH: %s\n", [[self fetchStatus:status] UTF8String]);
if (status == errSecItemNotFound) return NULL;
return outDictionary;
}
You don't need to disable ARC for this; you just need to declare the query's result as a
CFDictionaryRef
, then cast it to anNSDictionary
after the call.Couple of remarks:
__bridge_retained
to make sure ARC does not release and deallocate the object while we are working with it. This kind of bridging cast retains the object, so to prevent leaks, it must be followed with a correspondingCFRelease
somewhere.SecItemCopyMatching
definitely will not release the query for us, so if we use a retained bridge, then we need to release the resulting Core Foundation object ourself. (Which we do in line 4.)SecItemCopyMatching
has created its result with a retain count of 1, which we are responsible for releasing. (We know this because it has "Copy" in its name.)__bridge_transfer
lets ARC know about this responsibility, so it will be able to do it for us automatically.NSMutableDictionary
; that's just wrong. Also, it is against general Cocoa style conventions thatbuildSearchQuery
returns anNSMutableDictionary
. SimpleNSDictionary
s would work fine in both cases.The rule of thumb here is that
__bridge_retained
needs to be followed by aCFRelease
, while the result from a "Copy" or "Create" function must be cast into Cocoa-land using__bridge_transfer
.Method 3: Let ARC do the heavy lifting (or a combination of method 1 and method 2):
SecItemCopyMatching doesn't need to own the incoming dictionary, so __bridge is adequate and then ARC continues to manage the lifetime of query.
And by transferring ownership of the result to arc, it will manage the lifetime of result as well and free us of the need to remember to CFRelease it on all code paths.
Method 2: When you use it once, why need a retaining or transfering? example from bottom work at glance for me, testing, debuging(memleaks) all pass :) Only one thing did leak unreleased autoretained variable when key will be found (1)