/* * Copyright (c) 2008, 2009, 2010, 2011, 2012 * Jonathan Schleifer <js@webkeks.org> * * 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" #include <stdlib.h> #import "OFAutoreleasePool.h" #import "OFArray.h" #import "OFNotImplementedException.h" #import "macros.h" #ifndef OF_COMPILER_TLS # import "threading.h" # import "OFInitializationFailedException.h" #endif #import "autorelease.h" #define MAX_CACHE_SIZE 0x20 #ifdef OF_COMPILER_TLS static __thread void *first = NULL; static __thread OFAutoreleasePool **cache = NULL; #else static of_tlskey_t firstKey, cacheKey; #endif @implementation OFAutoreleasePool #ifndef OF_COMPILER_TLS + (void)initialize { if (self != [OFAutoreleasePool class]) return; if (!of_tlskey_new(&firstKey) || !of_tlskey_new(&cacheKey)) @throw [OFInitializationFailedException exceptionWithClass: self]; } #endif + alloc { #ifndef OF_COMPILER_TLS OFAutoreleasePool **cache = of_tlskey_get(cacheKey); #endif if (cache != NULL) { unsigned i; for (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 { #ifndef OF_COMPILER_TLS void *first = of_tlskey_get(firstKey); #endif if (first == NULL) [[OFAutoreleasePool alloc] init]; return _objc_rootAutorelease(object); } + (void)_releaseAll { #ifndef OF_COMPILER_TLS void *first = of_tlskey_get(firstKey); #endif objc_autoreleasePoolPop(first); } - init { self = [super init]; @try { #ifndef OF_COMPILER_TLS void *first = of_tlskey_get(firstKey); #endif pool = objc_autoreleasePoolPush(); if (first == NULL) #ifdef OF_COMPILER_TLS first = pool; #else OF_ENSURE(of_tlskey_set(firstKey, pool)); #endif _objc_rootAutorelease(self); } @catch (id e) { [self release]; @throw e; } return self; } - (void)releaseObjects { ignoreRelease = YES; objc_autoreleasePoolPop(pool); pool = objc_autoreleasePoolPush(); _objc_rootAutorelease(self); ignoreRelease = NO; } - (void)release { [self dealloc]; } - (void)drain { [self dealloc]; } - (void)dealloc { #ifndef OF_COMPILER_TLS OFAutoreleasePool **cache = of_tlskey_get(cacheKey); #endif if (ignoreRelease) return; ignoreRelease = YES; #ifdef OF_COMPILER_TLS if (first == pool) first = NULL; #else if (of_tlskey_get(firstKey) == pool) OF_ENSURE(of_tlskey_set(firstKey, NULL)); #endif objc_autoreleasePoolPop(pool); if (cache == NULL) { cache = calloc(sizeof(OFAutoreleasePool*), MAX_CACHE_SIZE); #ifndef OF_COMPILER_TLS if (!of_tlskey_set(cacheKey, cache)) { free(cache); cache = NULL; } #endif } if (cache != NULL) { unsigned i; for (i = 0; i < MAX_CACHE_SIZE; i++) { if (cache[i] == NULL) { pool = NULL; ignoreRelease = NO; cache[i] = self; return; } } } [super dealloc]; } - retain { @throw [OFNotImplementedException exceptionWithClass: [self class] selector: _cmd]; } - autorelease { @throw [OFNotImplementedException exceptionWithClass: [self class] selector: _cmd]; } @end