2011-03-11 08:14:04 +00:00
|
|
|
/** Definitions for block support for GNUStep
|
|
|
|
Copyright (C) 2011 Free Software Foundation, Inc.
|
|
|
|
|
|
|
|
This file is part of the GNUstep Base Library.
|
|
|
|
|
|
|
|
This library is free software; you can redistribute it and/or
|
|
|
|
modify it under the terms of the GNU Lesser General Public
|
|
|
|
License as published by the Free Software Foundation; either
|
|
|
|
version 2 of the License, or (at your option) any later version.
|
2012-03-26 14:47:07 +00:00
|
|
|
|
2011-03-11 08:14:04 +00:00
|
|
|
This library is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
Library General Public License for more details.
|
2012-03-26 14:47:07 +00:00
|
|
|
|
2011-03-11 08:14:04 +00:00
|
|
|
You should have received a copy of the GNU Lesser General Public
|
|
|
|
License along with this library; if not, write to the Free
|
2024-11-07 13:37:59 +00:00
|
|
|
Software Foundation, Inc., 31 Milk Street #960789 Boston, MA 02196 USA.
|
2011-03-11 08:14:04 +00:00
|
|
|
|
2012-03-26 14:47:07 +00:00
|
|
|
*/
|
2011-03-11 08:14:04 +00:00
|
|
|
|
|
|
|
#ifndef __GSBlocks_h_GNUSTEP_BASE_INCLUDE
|
|
|
|
#define __GSBlocks_h_GNUSTEP_BASE_INCLUDE
|
Added GSBlocks header. This defines two macros:
DEFINE_BLOCK_TYPE() defines a new block type.
CALL_BLOCK() calls a block.
These use the compiler's native support for blocks if it exists, but otherwise implement the lower-level version. Note that the structure defined by the fall-back versions is NOT a complete definition of the block layout, and copies of this structure should never be instantiated.
This does not allow non-blocks-aware compiles (like GCC) to create blocks, but it does allow them to use blocks. This means that we can implement things like NSArray's -enumerateObjectsUsingBlock: without breaking GCC compatibility. To do so, you'd first need to declare a block type in the header, like this:
DEFINE_BLOCK_TYPE(GSEnumeratorBlock, void, id, NSUInteger, BOOL*);
Then declare the method like this:
- (void)enumerateObjectsUsingBlock: (GSEnumeratorBlock)aBlock;
Finally, implement it like this:
- (void)enumerateObjectsUsingBlock: (GSEnumeratorBlock)aBlock
{
NSUInteger count = 0;
BOOL shouldStop = NO;
for (obj in self)
{
CALL_BLOCK(aBlock, obj, count++, &shouldStop);
if (shouldStop)
{
return;
}
}
}
If you compile this with clang -fblocks then it will work using Clang's native support for blocks. If you compile it with GCC or clang without -fblocks then it will use the other version. Note that this example uses fast enumeration, so will only compile with clang. A proper GNUstep implementation should have some fall-back code for primitive compilers that don't support this either.
git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/libs/base/trunk@29169 72102866-910b-0410-8b05-ffd578937521
2009-12-27 13:54:51 +00:00
|
|
|
|
|
|
|
/* Define the has_feature pseudo-macro for GCC. */
|
2012-03-26 14:47:07 +00:00
|
|
|
#ifndef __has_feature
|
Added GSBlocks header. This defines two macros:
DEFINE_BLOCK_TYPE() defines a new block type.
CALL_BLOCK() calls a block.
These use the compiler's native support for blocks if it exists, but otherwise implement the lower-level version. Note that the structure defined by the fall-back versions is NOT a complete definition of the block layout, and copies of this structure should never be instantiated.
This does not allow non-blocks-aware compiles (like GCC) to create blocks, but it does allow them to use blocks. This means that we can implement things like NSArray's -enumerateObjectsUsingBlock: without breaking GCC compatibility. To do so, you'd first need to declare a block type in the header, like this:
DEFINE_BLOCK_TYPE(GSEnumeratorBlock, void, id, NSUInteger, BOOL*);
Then declare the method like this:
- (void)enumerateObjectsUsingBlock: (GSEnumeratorBlock)aBlock;
Finally, implement it like this:
- (void)enumerateObjectsUsingBlock: (GSEnumeratorBlock)aBlock
{
NSUInteger count = 0;
BOOL shouldStop = NO;
for (obj in self)
{
CALL_BLOCK(aBlock, obj, count++, &shouldStop);
if (shouldStop)
{
return;
}
}
}
If you compile this with clang -fblocks then it will work using Clang's native support for blocks. If you compile it with GCC or clang without -fblocks then it will use the other version. Note that this example uses fast enumeration, so will only compile with clang. A proper GNUstep implementation should have some fall-back code for primitive compilers that don't support this either.
git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/libs/base/trunk@29169 72102866-910b-0410-8b05-ffd578937521
2009-12-27 13:54:51 +00:00
|
|
|
#define __has_feature(x) 0
|
|
|
|
#endif
|
|
|
|
|
2013-08-14 21:38:07 +00:00
|
|
|
#ifndef GCC_VERSION
|
|
|
|
#define GCC_VERSION (__GNUC__ * 1000 + __GNUC_MINOR__)
|
|
|
|
#endif
|
|
|
|
|
Added GSBlocks header. This defines two macros:
DEFINE_BLOCK_TYPE() defines a new block type.
CALL_BLOCK() calls a block.
These use the compiler's native support for blocks if it exists, but otherwise implement the lower-level version. Note that the structure defined by the fall-back versions is NOT a complete definition of the block layout, and copies of this structure should never be instantiated.
This does not allow non-blocks-aware compiles (like GCC) to create blocks, but it does allow them to use blocks. This means that we can implement things like NSArray's -enumerateObjectsUsingBlock: without breaking GCC compatibility. To do so, you'd first need to declare a block type in the header, like this:
DEFINE_BLOCK_TYPE(GSEnumeratorBlock, void, id, NSUInteger, BOOL*);
Then declare the method like this:
- (void)enumerateObjectsUsingBlock: (GSEnumeratorBlock)aBlock;
Finally, implement it like this:
- (void)enumerateObjectsUsingBlock: (GSEnumeratorBlock)aBlock
{
NSUInteger count = 0;
BOOL shouldStop = NO;
for (obj in self)
{
CALL_BLOCK(aBlock, obj, count++, &shouldStop);
if (shouldStop)
{
return;
}
}
}
If you compile this with clang -fblocks then it will work using Clang's native support for blocks. If you compile it with GCC or clang without -fblocks then it will use the other version. Note that this example uses fast enumeration, so will only compile with clang. A proper GNUstep implementation should have some fall-back code for primitive compilers that don't support this either.
git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/libs/base/trunk@29169 72102866-910b-0410-8b05-ffd578937521
2009-12-27 13:54:51 +00:00
|
|
|
#if __has_feature(blocks)
|
2011-03-11 08:14:04 +00:00
|
|
|
|
2012-03-27 09:30:12 +00:00
|
|
|
#define BLOCK_SCOPE __block
|
Added GSBlocks header. This defines two macros:
DEFINE_BLOCK_TYPE() defines a new block type.
CALL_BLOCK() calls a block.
These use the compiler's native support for blocks if it exists, but otherwise implement the lower-level version. Note that the structure defined by the fall-back versions is NOT a complete definition of the block layout, and copies of this structure should never be instantiated.
This does not allow non-blocks-aware compiles (like GCC) to create blocks, but it does allow them to use blocks. This means that we can implement things like NSArray's -enumerateObjectsUsingBlock: without breaking GCC compatibility. To do so, you'd first need to declare a block type in the header, like this:
DEFINE_BLOCK_TYPE(GSEnumeratorBlock, void, id, NSUInteger, BOOL*);
Then declare the method like this:
- (void)enumerateObjectsUsingBlock: (GSEnumeratorBlock)aBlock;
Finally, implement it like this:
- (void)enumerateObjectsUsingBlock: (GSEnumeratorBlock)aBlock
{
NSUInteger count = 0;
BOOL shouldStop = NO;
for (obj in self)
{
CALL_BLOCK(aBlock, obj, count++, &shouldStop);
if (shouldStop)
{
return;
}
}
}
If you compile this with clang -fblocks then it will work using Clang's native support for blocks. If you compile it with GCC or clang without -fblocks then it will use the other version. Note that this example uses fast enumeration, so will only compile with clang. A proper GNUstep implementation should have some fall-back code for primitive compilers that don't support this either.
git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/libs/base/trunk@29169 72102866-910b-0410-8b05-ffd578937521
2009-12-27 13:54:51 +00:00
|
|
|
/**
|
|
|
|
* Defines a block type. Will work whether or not the compiler natively
|
|
|
|
* supports blocks.
|
|
|
|
*/
|
|
|
|
#define DEFINE_BLOCK_TYPE(name, retTy, argTys, ...) \
|
|
|
|
typedef retTy(^name)(argTys, ## __VA_ARGS__)
|
2012-03-27 09:30:12 +00:00
|
|
|
|
|
|
|
#define DEFINE_BLOCK_TYPE_NO_ARGS(name, retTy) \
|
|
|
|
typedef retTy(^name)()
|
|
|
|
|
Added GSBlocks header. This defines two macros:
DEFINE_BLOCK_TYPE() defines a new block type.
CALL_BLOCK() calls a block.
These use the compiler's native support for blocks if it exists, but otherwise implement the lower-level version. Note that the structure defined by the fall-back versions is NOT a complete definition of the block layout, and copies of this structure should never be instantiated.
This does not allow non-blocks-aware compiles (like GCC) to create blocks, but it does allow them to use blocks. This means that we can implement things like NSArray's -enumerateObjectsUsingBlock: without breaking GCC compatibility. To do so, you'd first need to declare a block type in the header, like this:
DEFINE_BLOCK_TYPE(GSEnumeratorBlock, void, id, NSUInteger, BOOL*);
Then declare the method like this:
- (void)enumerateObjectsUsingBlock: (GSEnumeratorBlock)aBlock;
Finally, implement it like this:
- (void)enumerateObjectsUsingBlock: (GSEnumeratorBlock)aBlock
{
NSUInteger count = 0;
BOOL shouldStop = NO;
for (obj in self)
{
CALL_BLOCK(aBlock, obj, count++, &shouldStop);
if (shouldStop)
{
return;
}
}
}
If you compile this with clang -fblocks then it will work using Clang's native support for blocks. If you compile it with GCC or clang without -fblocks then it will use the other version. Note that this example uses fast enumeration, so will only compile with clang. A proper GNUstep implementation should have some fall-back code for primitive compilers that don't support this either.
git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/libs/base/trunk@29169 72102866-910b-0410-8b05-ffd578937521
2009-12-27 13:54:51 +00:00
|
|
|
/**
|
|
|
|
* Calls a block. Works irrespective of whether the compiler supports blocks.
|
|
|
|
*/
|
2021-11-17 13:35:49 +00:00
|
|
|
#define CALL_NON_NULL_BLOCK(block, args, ...) block(args, ## __VA_ARGS__)
|
2011-03-11 08:14:04 +00:00
|
|
|
|
2012-03-26 14:47:07 +00:00
|
|
|
/**
|
|
|
|
* Calls a block without arguments.
|
|
|
|
*/
|
2021-11-17 13:35:49 +00:00
|
|
|
#define CALL_NON_NULL_BLOCK_NO_ARGS(block) block()
|
Added GSBlocks header. This defines two macros:
DEFINE_BLOCK_TYPE() defines a new block type.
CALL_BLOCK() calls a block.
These use the compiler's native support for blocks if it exists, but otherwise implement the lower-level version. Note that the structure defined by the fall-back versions is NOT a complete definition of the block layout, and copies of this structure should never be instantiated.
This does not allow non-blocks-aware compiles (like GCC) to create blocks, but it does allow them to use blocks. This means that we can implement things like NSArray's -enumerateObjectsUsingBlock: without breaking GCC compatibility. To do so, you'd first need to declare a block type in the header, like this:
DEFINE_BLOCK_TYPE(GSEnumeratorBlock, void, id, NSUInteger, BOOL*);
Then declare the method like this:
- (void)enumerateObjectsUsingBlock: (GSEnumeratorBlock)aBlock;
Finally, implement it like this:
- (void)enumerateObjectsUsingBlock: (GSEnumeratorBlock)aBlock
{
NSUInteger count = 0;
BOOL shouldStop = NO;
for (obj in self)
{
CALL_BLOCK(aBlock, obj, count++, &shouldStop);
if (shouldStop)
{
return;
}
}
}
If you compile this with clang -fblocks then it will work using Clang's native support for blocks. If you compile it with GCC or clang without -fblocks then it will use the other version. Note that this example uses fast enumeration, so will only compile with clang. A proper GNUstep implementation should have some fall-back code for primitive compilers that don't support this either.
git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/libs/base/trunk@29169 72102866-910b-0410-8b05-ffd578937521
2009-12-27 13:54:51 +00:00
|
|
|
#else
|
2010-02-05 20:57:36 +00:00
|
|
|
|
2011-03-11 08:14:04 +00:00
|
|
|
/* Fall-back versions for when the compiler doesn't have native blocks support.
|
|
|
|
*/
|
2010-02-05 20:57:36 +00:00
|
|
|
#if (GCC_VERSION >= 3000)
|
|
|
|
|
Added GSBlocks header. This defines two macros:
DEFINE_BLOCK_TYPE() defines a new block type.
CALL_BLOCK() calls a block.
These use the compiler's native support for blocks if it exists, but otherwise implement the lower-level version. Note that the structure defined by the fall-back versions is NOT a complete definition of the block layout, and copies of this structure should never be instantiated.
This does not allow non-blocks-aware compiles (like GCC) to create blocks, but it does allow them to use blocks. This means that we can implement things like NSArray's -enumerateObjectsUsingBlock: without breaking GCC compatibility. To do so, you'd first need to declare a block type in the header, like this:
DEFINE_BLOCK_TYPE(GSEnumeratorBlock, void, id, NSUInteger, BOOL*);
Then declare the method like this:
- (void)enumerateObjectsUsingBlock: (GSEnumeratorBlock)aBlock;
Finally, implement it like this:
- (void)enumerateObjectsUsingBlock: (GSEnumeratorBlock)aBlock
{
NSUInteger count = 0;
BOOL shouldStop = NO;
for (obj in self)
{
CALL_BLOCK(aBlock, obj, count++, &shouldStop);
if (shouldStop)
{
return;
}
}
}
If you compile this with clang -fblocks then it will work using Clang's native support for blocks. If you compile it with GCC or clang without -fblocks then it will use the other version. Note that this example uses fast enumeration, so will only compile with clang. A proper GNUstep implementation should have some fall-back code for primitive compilers that don't support this either.
git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/libs/base/trunk@29169 72102866-910b-0410-8b05-ffd578937521
2009-12-27 13:54:51 +00:00
|
|
|
#define DEFINE_BLOCK_TYPE(name, retTy, argTys, ...) \
|
2011-03-11 08:14:04 +00:00
|
|
|
typedef struct {\
|
|
|
|
void *isa;\
|
|
|
|
int flags;\
|
|
|
|
int reserved;\
|
|
|
|
retTy (*invoke)(void*, argTys, ## __VA_ARGS__);\
|
|
|
|
} *name
|
|
|
|
|
2013-08-15 11:04:06 +00:00
|
|
|
#define DEFINE_BLOCK_TYPE_NO_ARGS(name, retTy) \
|
|
|
|
typedef struct {\
|
|
|
|
void *isa;\
|
|
|
|
int flags;\
|
|
|
|
int reserved;\
|
|
|
|
retTy (*invoke)(void*);\
|
|
|
|
} *name
|
|
|
|
|
2021-11-17 13:35:49 +00:00
|
|
|
#define CALL_NON_NULL_BLOCK(block, args, ...) block->invoke(block, args, ## __VA_ARGS__)
|
2010-02-05 20:57:36 +00:00
|
|
|
|
2021-11-17 13:35:49 +00:00
|
|
|
#define CALL_NON_NULL_BLOCK_NO_ARGS(block) block->invoke(block)
|
2013-08-15 07:09:09 +00:00
|
|
|
#define BLOCK_SCOPE
|
2012-03-26 14:47:07 +00:00
|
|
|
|
2010-02-05 20:57:36 +00:00
|
|
|
#else /* GCC_VERSION >= 3000 */
|
|
|
|
|
2013-08-14 21:34:31 +00:00
|
|
|
#define DEFINE_BLOCK_TYPE(name, retTy, argTys...) \
|
2011-03-11 08:14:04 +00:00
|
|
|
typedef struct {\
|
|
|
|
void *isa;\
|
|
|
|
int flags;\
|
|
|
|
int reserved;\
|
2013-08-14 21:34:31 +00:00
|
|
|
retTy (*invoke)(void*, argTys);\
|
2011-03-11 08:14:04 +00:00
|
|
|
} *name
|
2010-02-05 20:57:36 +00:00
|
|
|
|
2012-03-27 09:30:12 +00:00
|
|
|
#define DEFINE_BLOCK_TYPE_NO_ARGS(name, retTy) \
|
|
|
|
typedef struct {\
|
|
|
|
void *isa;\
|
|
|
|
int flags;\
|
|
|
|
int reserved;\
|
|
|
|
retTy (*invoke)(void*);\
|
|
|
|
} *name
|
2010-02-05 20:57:36 +00:00
|
|
|
|
2012-03-27 09:30:12 +00:00
|
|
|
|
2021-11-17 13:35:49 +00:00
|
|
|
#define CALL_NON_NULL_BLOCK(block, args...) block->invoke(block, args)
|
|
|
|
#define CALL_NON_NULL_BLOCK_NO_ARGS(block) block->invoke(block)
|
2012-03-27 09:30:12 +00:00
|
|
|
#define BLOCK_SCOPE
|
2010-02-05 20:57:36 +00:00
|
|
|
#endif /* GCC_VERSION >= 3000 */
|
|
|
|
|
2011-03-11 08:14:04 +00:00
|
|
|
#endif /* __has_feature(blocks) */
|
|
|
|
|
2023-11-14 21:20:50 +00:00
|
|
|
#define CALL_BLOCK(block, args...) ({if (NULL != block) CALL_NON_NULL_BLOCK(block, args);})
|
|
|
|
#define CALL_BLOCK_RET(block, rettype, args...) ((NULL != block) ? (rettype)CALL_NON_NULL_BLOCK(block, args) : (rettype)0)
|
|
|
|
|
|
|
|
#define CALL_BLOCK_NO_ARGS(block) ({if (NULL != block) CALL_NON_NULL_BLOCK_NO_ARGS(block);})
|
|
|
|
#define CALL_BLOCK_RET_NO_ARGS(block, rettype) ((NULL != block) ? (rettype)CALL_NON_NULL_BLOCK_NO_ARGS(block) : (rettype)0)
|
2021-11-17 13:35:49 +00:00
|
|
|
|
2017-12-02 11:16:30 +00:00
|
|
|
#if __has_include(<objc/blocks_runtime.h>)
|
|
|
|
# include <objc/blocks_runtime.h>
|
|
|
|
#else
|
|
|
|
|
2013-06-20 22:04:14 +00:00
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/**
|
|
|
|
* _Block_copy and _Block_release are weakly imported, but can be assumed
|
|
|
|
* to be available whenever a feature using blocks is accessed
|
|
|
|
* by an application.
|
|
|
|
*/
|
|
|
|
|
2015-09-22 08:26:10 +00:00
|
|
|
/* weak attributed supported only with ELF, MINGW is COFF */
|
|
|
|
#ifndef __MINGW32__
|
|
|
|
|
2017-11-18 16:13:21 +00:00
|
|
|
void *_Block_copy(const void *) __attribute__((weak));
|
|
|
|
void _Block_release(const void *) __attribute__((weak));
|
2013-06-20 22:04:14 +00:00
|
|
|
|
2015-09-22 08:26:10 +00:00
|
|
|
#endif /* __MINGW32__ */
|
|
|
|
|
2013-06-20 22:04:14 +00:00
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef Block_copy
|
2020-12-15 13:01:00 +00:00
|
|
|
# define Block_copy(x) ((__typeof__(x))_Block_copy((const void *)(x)))
|
2013-06-20 22:04:14 +00:00
|
|
|
#endif
|
|
|
|
#ifndef Block_release
|
2017-11-18 16:13:21 +00:00
|
|
|
# define Block_release(x) _Block_release((const void *)(x))
|
2013-06-20 22:04:14 +00:00
|
|
|
#endif
|
|
|
|
|
2017-12-02 11:16:30 +00:00
|
|
|
#endif /* __has_include(<objc/blocks_runtime.h>) */
|
|
|
|
|
2011-03-11 08:14:04 +00:00
|
|
|
#endif /* __GSBlocks_h_GNUSTEP_BASE_INCLUDE */
|
Added GSBlocks header. This defines two macros:
DEFINE_BLOCK_TYPE() defines a new block type.
CALL_BLOCK() calls a block.
These use the compiler's native support for blocks if it exists, but otherwise implement the lower-level version. Note that the structure defined by the fall-back versions is NOT a complete definition of the block layout, and copies of this structure should never be instantiated.
This does not allow non-blocks-aware compiles (like GCC) to create blocks, but it does allow them to use blocks. This means that we can implement things like NSArray's -enumerateObjectsUsingBlock: without breaking GCC compatibility. To do so, you'd first need to declare a block type in the header, like this:
DEFINE_BLOCK_TYPE(GSEnumeratorBlock, void, id, NSUInteger, BOOL*);
Then declare the method like this:
- (void)enumerateObjectsUsingBlock: (GSEnumeratorBlock)aBlock;
Finally, implement it like this:
- (void)enumerateObjectsUsingBlock: (GSEnumeratorBlock)aBlock
{
NSUInteger count = 0;
BOOL shouldStop = NO;
for (obj in self)
{
CALL_BLOCK(aBlock, obj, count++, &shouldStop);
if (shouldStop)
{
return;
}
}
}
If you compile this with clang -fblocks then it will work using Clang's native support for blocks. If you compile it with GCC or clang without -fblocks then it will use the other version. Note that this example uses fast enumeration, so will only compile with clang. A proper GNUstep implementation should have some fall-back code for primitive compilers that don't support this either.
git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/libs/base/trunk@29169 72102866-910b-0410-8b05-ffd578937521
2009-12-27 13:54:51 +00:00
|
|
|
|