/* * Copyright (c) 2008, 2009, 2010, 2011, 2012, 2013, 2014, 2015, 2016, 2017, * 2018 * Jonathan Schleifer * * All rights reserved. * * This file is part of ObjFW. It may be distributed under the terms of the * Q Public License 1.0, which can be found in the file LICENSE.QPL included in * the packaging of this file. * * Alternatively, it may be distributed under the terms of the GNU General * Public License, either version 2 or 3, which can be found in the file * LICENSE.GPLv2 or LICENSE.GPLv3 respectively included in the packaging of this * file. */ #include "config.h" #import "OFAutoreleasePool.h" #import "OFAutoreleasePool+Private.h" #if !defined(OF_HAVE_COMPILER_TLS) && defined(OF_HAVE_THREADS) # import "threading.h" # # import "OFInitializationFailedException.h" #endif #define MAX_CACHE_SIZE 0x20 #if defined(OF_HAVE_COMPILER_TLS) static thread_local OFAutoreleasePool **cache = NULL; #elif defined(OF_HAVE_THREADS) static of_tlskey_t cacheKey; #else static OFAutoreleasePool **cache = NULL; #endif @implementation OFAutoreleasePool #if !defined(OF_HAVE_COMPILER_TLS) && defined(OF_HAVE_THREADS) + (void)initialize { if (self != [OFAutoreleasePool class]) return; if (!of_tlskey_new(&cacheKey)) @throw [OFInitializationFailedException exceptionWithClass: self]; } #endif + (instancetype)alloc { #if !defined(OF_HAVE_COMPILER_TLS) && defined(OF_HAVE_THREADS) OFAutoreleasePool **cache = of_tlskey_get(cacheKey); #endif if (cache != NULL) { for (size_t i = 0; i < MAX_CACHE_SIZE; i++) { if (cache[i] != NULL) { OFAutoreleasePool *pool = cache[i]; cache[i] = NULL; return pool; } } } return [super alloc]; } + (id)addObject: (id)object { return _objc_rootAutorelease(object); } + (void)of_handleThreadTermination { #if !defined(OF_HAVE_COMPILER_TLS) && defined(OF_HAVE_THREADS) OFAutoreleasePool **cache = of_tlskey_get(cacheKey); #endif if (cache != NULL) { for (size_t i = 0; i < MAX_CACHE_SIZE; i++) [cache[i] of_super_dealloc]; free(cache); cache = NULL; } } - (instancetype)init { self = [super init]; @try { _pool = objc_autoreleasePoolPush(); _objc_rootAutorelease(self); } @catch (id e) { [self release]; @throw e; } return self; } - (void)releaseObjects { _ignoreRelease = true; objc_autoreleasePoolPop(_pool); _pool = objc_autoreleasePoolPush(); _objc_rootAutorelease(self); _ignoreRelease = false; } - (void)release { [self dealloc]; } - (void)drain { [self dealloc]; } - (void)of_super_dealloc { [super dealloc]; } - (void)dealloc { #if !defined(OF_HAVE_COMPILER_TLS) && defined(OF_HAVE_THREADS) OFAutoreleasePool **cache = of_tlskey_get(cacheKey); #endif if (_ignoreRelease) return; _ignoreRelease = true; objc_autoreleasePoolPop(_pool); if (cache == NULL) { cache = calloc(sizeof(OFAutoreleasePool *), MAX_CACHE_SIZE); #if !defined(OF_HAVE_COMPILER_TLS) && defined(OF_HAVE_THREADS) if (!of_tlskey_set(cacheKey, cache)) { free(cache); cache = NULL; } #endif } if (cache != NULL) { for (size_t i = 0; i < MAX_CACHE_SIZE; i++) { if (cache[i] == NULL) { _pool = NULL; _ignoreRelease = false; cache[i] = self; return; } } } [super dealloc]; } - (instancetype)retain { OF_UNRECOGNIZED_SELECTOR } - (instancetype)autorelease { OF_UNRECOGNIZED_SELECTOR } @end