mirror of
https://github.com/gnustep/libs-base.git
synced 2025-04-26 02:01:03 +00:00
* GSAtomic: Add prefix to macro definitions * NSKVOSupport: Import * NSKVOSupport: Add test cases * NSKVOSwizzling: Ugly C Rewrite * NSKeyValueObserving: Use old implementation as fallback * NSKeyValueObserving: Rename TypeEncodingCases header * NSKVOSupport: Fix new objects not being added to NSKeyValueChangeNew set on set mutation * NSKeyValueMutableSet: Fix will and didChange notifications for set operations * NSKeyValueMutableSet: Document Accessor Search Patterns * NSKVOSupport: Add toMany test * NSKeyValueCoding: Change notifications when changing value via setValue:forKey: * NSKVOSupport: Add more tests * NSKVOSupport: Do not wrap block in try/finally to avoid crash in windows * NSKVOSwizzling: use _alloca on Windows * NSKVOSupport: Do not autorelease newWithObservee: * NSKVOSupport: Do not leak Observee and TestFacade objects * Improve runtime detection in makefile * Add file extension of source file in GNUMakefile * NSKVOSupport: Remove @status comments * NSKVOSupport: Implement private notify method * NSUserDefaults: KVO Support and fix macOS incompatibilities * NSKeyValueObserving: Set old to null if nil * NSKeyValueObserving: Remove cached new value * NSMethodSignature: Add signature cache * NSKVOSupport: Remove ObjC2 features and mark tests failing on GCC as hopeful * Call class method instead of private _keyPathsForValuesAffectingValueForKey * Move _keyPathsForValuesAffectingValueForKey body into class method and statically construct empty NSSet * NSUserDefaults: Change notification should contain old value from other domains aswell * NSUserDefaults: Fetch new value from all domains * NSKVOInternal: Fixup filename in header * NSUserDefaults: Go through search list instead of only one domain in KVO change * Making indentation a bit less worse * Add NSUserDefaults KVO tests * NSKVOSupport: NSUserDefaults test small fixes * Add autoreleasepool * NSUserDefaults: Only emit change notifications if value changed * Avoid compiler warnings and tidy some of the whitespace/formatting --------- Co-authored-by: Frederik Seiffert <frederik@algoriddim.com> Co-authored-by: rfm <richardfrithmacdonald@gmail.com> Co-authored-by: rfm <rfm@gnu.org>
72 lines
2.1 KiB
C
72 lines
2.1 KiB
C
/*
|
|
* Provides atomic load and store functions using either native C11 atomic
|
|
* types and operations if available, or otherwise using fallback
|
|
* implementations (e.g. with GCC where stdatomic.h is not useable from
|
|
* Objective-C).
|
|
*
|
|
* Adopted from FreeBSD's stdatomic.h.
|
|
*/
|
|
#ifndef _GSAtomic_h_
|
|
#define _GSAtomic_h_
|
|
|
|
#ifndef __has_extension
|
|
#define __has_extension(x) 0
|
|
#endif
|
|
|
|
#if __has_extension(c_atomic) || __has_extension(cxx_atomic)
|
|
|
|
/*
|
|
* Use native C11 atomic operations. _Atomic() should be defined by the
|
|
* compiler.
|
|
*/
|
|
#define gs_atomic_load_explicit(object, order) \
|
|
__c11_atomic_load(object, order)
|
|
#define gs_atomic_store_explicit(object, desired, order) \
|
|
__c11_atomic_store(object, desired, order)
|
|
|
|
#else
|
|
|
|
/*
|
|
* No native support for _Atomic(). Place object in structure to prevent
|
|
* most forms of direct non-atomic access.
|
|
*/
|
|
#define _Atomic(T) struct { T volatile __val; }
|
|
#if __has_builtin(__sync_swap)
|
|
/* Clang provides a full-barrier atomic exchange - use it if available. */
|
|
#define gs_atomic_exchange_explicit(object, desired, order) \
|
|
((void)(order), __sync_swap(&(object)->__val, desired))
|
|
#else
|
|
/*
|
|
* __sync_lock_test_and_set() is only an acquire barrier in theory (although in
|
|
* practice it is usually a full barrier) so we need an explicit barrier before
|
|
* it.
|
|
*/
|
|
#define gs_atomic_exchange_explicit(object, desired, order) \
|
|
__extension__ ({ \
|
|
__typeof__(object) __o = (object); \
|
|
__typeof__(desired) __d = (desired); \
|
|
(void)(order); \
|
|
__sync_synchronize(); \
|
|
__sync_lock_test_and_set(&(__o)->__val, __d); \
|
|
})
|
|
#endif
|
|
#define gs_atomic_load_explicit(object, order) \
|
|
((void)(order), __sync_fetch_and_add(&(object)->__val, 0))
|
|
#define gs_atomic_store_explicit(object, desired, order) \
|
|
((void)gs_atomic_exchange_explicit(object, desired, order))
|
|
|
|
#endif
|
|
|
|
#ifndef __ATOMIC_SEQ_CST
|
|
#define __ATOMIC_SEQ_CST 5
|
|
#endif
|
|
|
|
/*
|
|
* Convenience functions.
|
|
*/
|
|
#define gs_atomic_load(object) \
|
|
gs_atomic_load_explicit(object, __ATOMIC_SEQ_CST)
|
|
#define gs_atomic_store(object, desired) \
|
|
gs_atomic_store_explicit(object, desired, __ATOMIC_SEQ_CST)
|
|
|
|
#endif // _GSAtomic_h_
|