mirror of https://github.com/oxen-io/session-ios
				
				
				
			
			You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
	
	
		
			211 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Objective-C
		
	
			
		
		
	
	
			211 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Objective-C
		
	
| //
 | |
| //  Copyright (c) 2018 Open Whisper Systems. All rights reserved.
 | |
| //
 | |
| 
 | |
| #import "OWSLogs.h"
 | |
| 
 | |
| NS_ASSUME_NONNULL_BEGIN
 | |
| 
 | |
| #ifndef OWSAssert
 | |
| 
 | |
| #define CONVERT_TO_STRING(X) #X
 | |
| #define CONVERT_EXPR_TO_STRING(X) CONVERT_TO_STRING(X)
 | |
| 
 | |
| #ifdef DEBUG
 | |
| 
 | |
| #define USE_ASSERTS
 | |
| 
 | |
| // OWSAssertDebug() and OWSFailDebug() should be used in Obj-C methods.
 | |
| // OWSCAssertDebug() and OWSCFailDebug() should be used in free functions.
 | |
| 
 | |
| #define OWSAssertDebug(X)                                                                                              \
 | |
|     do {                                                                                                               \
 | |
|         if (!(X)) {                                                                                                    \
 | |
|             OWSLogError(@"Assertion failed: %s", CONVERT_EXPR_TO_STRING(X));                                           \
 | |
|             OWSLogFlush();                                                                                             \
 | |
|             NSAssert(0, @"Assertion failed: %s", CONVERT_EXPR_TO_STRING(X));                                           \
 | |
|         }                                                                                                              \
 | |
|     } while (NO)
 | |
| 
 | |
| #define OWSCAssertDebug(X)                                                                                             \
 | |
|     do {                                                                                                               \
 | |
|         if (!(X)) {                                                                                                    \
 | |
|             OWSLogError(@"Assertion failed: %s", CONVERT_EXPR_TO_STRING(X));                                           \
 | |
|             OWSLogFlush();                                                                                             \
 | |
|             NSCAssert(0, @"Assertion failed: %s", CONVERT_EXPR_TO_STRING(X));                                          \
 | |
|         }                                                                                                              \
 | |
|     } while (NO)
 | |
| 
 | |
| #define OWSFailWithoutLogging(message, ...)                                                                            \
 | |
|     do {                                                                                                               \
 | |
|         NSString *formattedMessage = [NSString stringWithFormat:message, ##__VA_ARGS__];                               \
 | |
|         NSAssert(0, formattedMessage);                                                                                 \
 | |
|     } while (NO)
 | |
| 
 | |
| #define OWSCFailWithoutLogging(message, ...)                                                                           \
 | |
|     do {                                                                                                               \
 | |
|         NSString *formattedMessage = [NSString stringWithFormat:message, ##__VA_ARGS__];                               \
 | |
|         NSCAssert(0, formattedMessage);                                                                                \
 | |
|     } while (NO)
 | |
| 
 | |
| #define OWSFailNoFormat(message)                                                                                       \
 | |
|     do {                                                                                                               \
 | |
|         OWSLogError(@"%@", message);                                                                                   \
 | |
|         OWSLogFlush();                                                                                                 \
 | |
|         NSAssert(0, message);                                                                                          \
 | |
|     } while (NO)
 | |
| 
 | |
| #define OWSCFailNoFormat(message)                                                                                      \
 | |
|     do {                                                                                                               \
 | |
|         OWSLogError(@"%@", message);                                                                                   \
 | |
|         OWSLogFlush();                                                                                                 \
 | |
|         NSCAssert(0, message);                                                                                         \
 | |
|     } while (NO)
 | |
| 
 | |
| #else
 | |
| 
 | |
| #define OWSAssertDebug(X)
 | |
| #define OWSCAssertDebug(X)
 | |
| #define OWSFailWithoutLogging(message, ...)
 | |
| #define OWSCFailWithoutLogging(message, ...)
 | |
| #define OWSFailNoFormat(X)
 | |
| #define OWSCFailNoFormat(X)
 | |
| 
 | |
| #endif
 | |
| 
 | |
| #endif
 | |
| 
 | |
| // Like OWSAssertDebug, but will fail in production, terminating the app
 | |
| #define OWSAssert(X)                                                                                                   \
 | |
|     do {                                                                                                               \
 | |
|         if (!(X)) {                                                                                                    \
 | |
|             OWSFail(@"Assertion failed: %s", CONVERT_EXPR_TO_STRING(X));                                               \
 | |
|         }                                                                                                              \
 | |
|     } while (NO)
 | |
| 
 | |
| #define OWSCAssert(X)                                                                                                  \
 | |
|     do {                                                                                                               \
 | |
|         if (!(X)) {                                                                                                    \
 | |
|             OWSCFail(@"Assertion failed: %s", CONVERT_EXPR_TO_STRING(X));                                              \
 | |
|         }                                                                                                              \
 | |
|     } while (NO)
 | |
| 
 | |
| #define OWSAbstractMethod() OWSFail(@"Method needs to be implemented by subclasses.")
 | |
| 
 | |
| // This macro is intended for use in Objective-C.
 | |
| #define OWSAssertIsOnMainThread() OWSCAssertDebug([NSThread isMainThread])
 | |
| 
 | |
| #define OWSFailDebug(_messageFormat, ...)                                                                              \
 | |
|     do {                                                                                                               \
 | |
|         OWSLogError(_messageFormat, ##__VA_ARGS__);                                                                    \
 | |
|         OWSLogFlush();                                                                                                 \
 | |
|         OWSFailWithoutLogging(_messageFormat, ##__VA_ARGS__);                                                          \
 | |
|     } while (0)
 | |
| 
 | |
| #define OWSCFailDebug(_messageFormat, ...)                                                                             \
 | |
|     do {                                                                                                               \
 | |
|         OWSLogError(_messageFormat, ##__VA_ARGS__);                                                                    \
 | |
|         OWSLogFlush();                                                                                                 \
 | |
|         OWSCFailWithoutLogging(_messageFormat, ##__VA_ARGS__);                                                         \
 | |
|     } while (NO)
 | |
| 
 | |
| void SwiftExit(NSString *message, const char *file, const char *function, int line);
 | |
| 
 | |
| #define OWSFail(_messageFormat, ...)                                                                                   \
 | |
|     do {                                                                                                               \
 | |
|         OWSFailDebug(_messageFormat, ##__VA_ARGS__);                                                                   \
 | |
|                                                                                                                        \
 | |
|         NSString *_message = [NSString stringWithFormat:_messageFormat, ##__VA_ARGS__];                                \
 | |
|         SwiftExit(_message, __FILE__, __PRETTY_FUNCTION__, __LINE__);                                                  \
 | |
|     } while (0)
 | |
| 
 | |
| #define OWSCFail(_messageFormat, ...)                                                                                  \
 | |
|     do {                                                                                                               \
 | |
|         OWSCFailDebug(_messageFormat, ##__VA_ARGS__);                                                                  \
 | |
|                                                                                                                        \
 | |
|         NSString *_message = [NSString stringWithFormat:_messageFormat, ##__VA_ARGS__];                                \
 | |
|         SwiftExit(_message, __FILE__, __PRETTY_FUNCTION__, __LINE__);                                                  \
 | |
|     } while (NO)
 | |
| 
 | |
| // Avoids Clang analyzer warning:
 | |
| //   Value stored to 'x' during it's initialization is never read
 | |
| #define SUPPRESS_DEADSTORE_WARNING(x)                                                                                  \
 | |
|     do {                                                                                                               \
 | |
|         (void)x;                                                                                                       \
 | |
|     } while (0)
 | |
| 
 | |
| __attribute__((annotate("returns_localized_nsstring"))) static inline NSString *LocalizationNotNeeded(NSString *s)
 | |
| {
 | |
|     return s;
 | |
| }
 | |
| 
 | |
| #define OWSGuardWithException(X, ExceptionName)                                                                        \
 | |
|     do {                                                                                                               \
 | |
|         if (!(X)) {                                                                                                    \
 | |
|             OWSRaiseException(ExceptionName, @"Guard failed: %s", CONVERT_EXPR_TO_STRING(X));                          \
 | |
|         }                                                                                                              \
 | |
|     } while (NO)
 | |
| 
 | |
| #define OWSRaiseException(name, formatParam, ...)                                                                      \
 | |
|     do {                                                                                                               \
 | |
|         OWSLogError(@"Exception: %@ %@", name, [NSString stringWithFormat:formatParam, ##__VA_ARGS__]);                \
 | |
|         OWSLogFlush();                                                                                                 \
 | |
|         @throw [NSException exceptionWithName:name                                                                     \
 | |
|                                        reason:[NSString stringWithFormat:formatParam, ##__VA_ARGS__]                   \
 | |
|                                      userInfo:nil];                                                                    \
 | |
|     } while (NO)
 | |
| 
 | |
| #define OWSRaiseExceptionWithUserInfo(name, userInfoParam, formatParam, ...)                                           \
 | |
|     do {                                                                                                               \
 | |
|         OWSLogError(                                                                                                   \
 | |
|             @"Exception: %@ %@ %@", name, userInfoParam, [NSString stringWithFormat:formatParam, ##__VA_ARGS__]);      \
 | |
|         OWSLogFlush();                                                                                                 \
 | |
|         @throw [NSException exceptionWithName:name                                                                     \
 | |
|                                        reason:[NSString stringWithFormat:formatParam, ##__VA_ARGS__]                   \
 | |
|                                      userInfo:userInfoParam];                                                          \
 | |
|     } while (NO)
 | |
| 
 | |
| 
 | |
| // UI JANK
 | |
| //
 | |
| // In pursuit of smooth UI, we want to continue moving blocking operations off the main thread.
 | |
| // Add `OWSJanksUI` in code paths that shouldn't be called on the main thread.
 | |
| // Because we have pervasively broken this tenant, enabling it by default would be too disruptive
 | |
| // but it's helpful while unjanking and maybe someday we can have it enabled by default.
 | |
| //#define DEBUG_UI_JANK 1
 | |
| 
 | |
| #ifdef DEBUG
 | |
| #ifdef DEBUG_UI_JANK
 | |
| #define OWSJanksUI()                                                                                                   \
 | |
|     do {                                                                                                               \
 | |
|         OWSAssertDebug(![NSThread isMainThread])                                                                       \
 | |
|     } while (NO)
 | |
| #endif
 | |
| #endif
 | |
| 
 | |
| #ifndef OWSJanksUI
 | |
| #define OWSJanksUI()
 | |
| #endif
 | |
| 
 | |
| #pragma mark - Overflow Math
 | |
| 
 | |
| #define ows_add_overflow(a, b, resultRef)                                                                              \
 | |
|     do {                                                                                                               \
 | |
|         BOOL _didOverflow = __builtin_add_overflow(a, b, resultRef);                                                   \
 | |
|         OWSAssert(!_didOverflow);                                                                                      \
 | |
|     } while (NO)
 | |
| 
 | |
| #define ows_sub_overflow(a, b, resultRef)                                                                              \
 | |
|     do {                                                                                                               \
 | |
|         BOOL _didOverflow = __builtin_sub_overflow(a, b, resultRef);                                                   \
 | |
|         OWSAssert(!_didOverflow);                                                                                      \
 | |
|     } while (NO)
 | |
| 
 | |
| #define ows_mul_overflow(a, b, resultRef)                                                                              \
 | |
|     do {                                                                                                               \
 | |
|         BOOL _didOverflow = __builtin_mul_overflow(a, b, resultRef);                                                   \
 | |
|         OWSAssert(!_didOverflow);                                                                                      \
 | |
|     } while (NO)
 | |
| 
 | |
| NS_ASSUME_NONNULL_END
 |