aboutsummaryrefslogtreecommitdiff
path: root/DebugUtils/GTMSynchronizationAsserts.h
blob: b0265658039579841617e4a84c67a3d05771e56e (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
//
//  GTMSynchronizationAsserts.h
//
//  Copyright 2016 Google Inc.
//
//  Licensed under the Apache License, Version 2.0 (the "License"); you may not
//  use this file except in compliance with the License.  You may obtain a copy
//  of the License at
//
//  http://www.apache.org/licenses/LICENSE-2.0
//
//  Unless required by applicable law or agreed to in writing, software
//  distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
//  WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.  See the
//  License for the specific language governing permissions and limitations under
//  the License.
//

#if !__has_feature(objc_arc)
#error "This file needs to be compiled with ARC enabled."
#endif

#import <Foundation/Foundation.h>

#import "GTMDefines.h"  // For _GTMDevAssert.

// Macros to monitor synchronization blocks in debug builds.
//
// These report problems using _GTMDevAssert, which may be defined by the
// project or by GTMDefines.h
//
// GTMMonitorSynchronized           Start monitoring a top-level-only @sync scope.
//                                  Asserts if already inside a monitored @sync scope.
// GTMMonitorRecursiveSynchronized  Start monitoring a top-level or recursive @sync
//                                  scope.
// GTMCheckSynchronized             Assert that the current execution is inside a monitored @sync
//                                  scope.
// GTMCheckNotSynchronized          Assert that the current execution is not inside a monitored
//                                  @sync scope.
//
// Example usage:
//
// - (void)myExternalMethod {
//   @synchronized(self) {
//     GTMMonitorSynchronized(self)
//
// - (void)myInternalMethod {
//   GTMCheckSynchronized(self);
//
// - (void)callMyCallbacks {
//   GTMCheckNotSynchronized(self);
//
// GTMCheckNotSynchronized is available for verifying the code isn't
// in a deadlockable @sync state, important when posting notifications and
// invoking callbacks.
//
// Don't use GTMCheckNotSynchronized immediately before a @sync scope; the
// normal recursiveness check of GTMMonitorSynchronized can catch those.

#if DEBUG

  #define __GTMMonitorSynchronizedVariableInner(varname, counter) \
      varname ## counter
  #define __GTMMonitorSynchronizedVariable(varname, counter)      \
      __GTMMonitorSynchronizedVariableInner(varname, counter)

  #define GTMMonitorSynchronized(obj)                                           \
      NS_VALID_UNTIL_END_OF_SCOPE id                                            \
        __GTMMonitorSynchronizedVariable(__monitor, __COUNTER__) =              \
        [[GTMSyncMonitorInternal alloc] initWithSynchronizationObject:obj       \
                                                    allowRecursive:NO           \
                                                     functionName:__func__]

  #define GTMMonitorRecursiveSynchronized(obj)                                  \
      NS_VALID_UNTIL_END_OF_SCOPE id                                            \
        __GTMMonitorSynchronizedVariable(__monitor, __COUNTER__) =              \
        [[GTMSyncMonitorInternal alloc] initWithSynchronizationObject:obj       \
                                                    allowRecursive:YES          \
                                                     functionName:__func__]

  #define GTMCheckSynchronized(obj) {                                           \
      _GTMDevAssert(                                                            \
          [GTMSyncMonitorInternal functionsHoldingSynchronizationOnObject:obj], \
          @"GTMCheckSynchronized(" #obj ") failed: not sync'd"                  \
          @" on " #obj " in %s. Call stack:\n%@",                               \
          __func__, [NSThread callStackSymbols]);                               \
      }

  #define GTMCheckNotSynchronized(obj) {                                       \
      _GTMDevAssert(                                                           \
        ![GTMSyncMonitorInternal functionsHoldingSynchronizationOnObject:obj], \
        @"GTMCheckNotSynchronized(" #obj ") failed: was sync'd"                \
        @" on " #obj " in %s by %@. Call stack:\n%@", __func__,                \
        [GTMSyncMonitorInternal functionsHoldingSynchronizationOnObject:obj],  \
        [NSThread callStackSymbols]);                                          \
      }

// GTMSyncMonitorInternal is a private class that keeps track of the
// beginning and end of synchronized scopes, relying on ARC to release
// it at the end of a scope.
//
// This class should not be used directly, but only via the
// GTMMonitorSynchronized macro.
@interface GTMSyncMonitorInternal : NSObject {
  NSValue *_objectKey;        // The synchronize target object.
  const char *_functionName;  // The function containing the monitored sync block.
}

- (instancetype)initWithSynchronizationObject:(id)object
                               allowRecursive:(BOOL)allowRecursive
                                 functionName:(const char *)functionName;
// Return the names of the functions that hold sync on the object, or nil if none.
+ (NSArray *)functionsHoldingSynchronizationOnObject:(id)object;
@end

#else

  // !DEBUG
  #define GTMMonitorSynchronized(obj) do { } while (0)
  #define GTMMonitorRecursiveSynchronized(obj) do { } while (0)
  #define GTMCheckSynchronized(obj) do { } while (0)
  #define GTMCheckNotSynchronized(obj) do { } while (0)

#endif  // DEBUG