| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240 |
- // Software License Agreement (BSD License)
- //
- // Copyright (c) 2010-2025, Deusty, LLC
- // All rights reserved.
- //
- // Redistribution and use of this software in source and binary forms,
- // with or without modification, are permitted provided that the following conditions are met:
- //
- // * Redistributions of source code must retain the above copyright notice,
- // this list of conditions and the following disclaimer.
- //
- // * Neither the name of Deusty nor the names of its contributors may be used
- // to endorse or promote products derived from this software without specific
- // prior written permission of Deusty, LLC.
- #if !__has_feature(objc_arc)
- #error This file must be compiled with ARC. Use -fobjc-arc flag (or convert project to ARC).
- #endif
- #import <pthread/pthread.h>
- #import <stdatomic.h>
- #import <sys/qos.h>
- #import <CocoaLumberjack/DDDispatchQueueLogFormatter.h>
- DDQualityOfServiceName const DDQualityOfServiceUserInteractive = @"UI";
- DDQualityOfServiceName const DDQualityOfServiceUserInitiated = @"IN";
- DDQualityOfServiceName const DDQualityOfServiceDefault = @"DF";
- DDQualityOfServiceName const DDQualityOfServiceUtility = @"UT";
- DDQualityOfServiceName const DDQualityOfServiceBackground = @"BG";
- DDQualityOfServiceName const DDQualityOfServiceUnspecified = @"UN";
- static DDQualityOfServiceName _qos_name(NSUInteger qos) {
- switch ((qos_class_t) qos) {
- case QOS_CLASS_USER_INTERACTIVE: return DDQualityOfServiceUserInteractive;
- case QOS_CLASS_USER_INITIATED: return DDQualityOfServiceUserInitiated;
- case QOS_CLASS_DEFAULT: return DDQualityOfServiceDefault;
- case QOS_CLASS_UTILITY: return DDQualityOfServiceUtility;
- case QOS_CLASS_BACKGROUND: return DDQualityOfServiceBackground;
- default: return DDQualityOfServiceUnspecified;
- }
- }
- #pragma mark - DDDispatchQueueLogFormatter
- @interface DDDispatchQueueLogFormatter () {
- NSDateFormatter *_dateFormatter; // Use [self stringFromDate]
- pthread_mutex_t _mutex;
- NSUInteger _minQueueLength; // _prefix == Only access via atomic property
- NSUInteger _maxQueueLength; // _prefix == Only access via atomic property
- NSMutableDictionary *_replacements; // _prefix == Only access from within spinlock
- }
- @end
- @implementation DDDispatchQueueLogFormatter
- - (instancetype)init {
- if ((self = [super init])) {
- _dateFormatter = [self createDateFormatter];
- pthread_mutex_init(&_mutex, NULL);
- _replacements = [[NSMutableDictionary alloc] init];
- // Set default replacements:
- _replacements[@"com.apple.main-thread"] = @"main";
- }
- return self;
- }
- - (instancetype)initWithMode:(DDDispatchQueueLogFormatterMode)mode {
- return [self init];
- }
- - (void)dealloc {
- pthread_mutex_destroy(&_mutex);
- }
- ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
- #pragma mark Configuration
- ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
- @synthesize minQueueLength = _minQueueLength;
- @synthesize maxQueueLength = _maxQueueLength;
- - (NSString *)replacementStringForQueueLabel:(NSString *)longLabel {
- NSString *result = nil;
- pthread_mutex_lock(&_mutex);
- {
- result = _replacements[longLabel];
- }
- pthread_mutex_unlock(&_mutex);
- return result;
- }
- - (void)setReplacementString:(NSString *)shortLabel forQueueLabel:(NSString *)longLabel {
- pthread_mutex_lock(&_mutex);
- {
- if (shortLabel) {
- _replacements[longLabel] = shortLabel;
- } else {
- [_replacements removeObjectForKey:longLabel];
- }
- }
- pthread_mutex_unlock(&_mutex);
- }
- ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
- #pragma mark DDLogFormatter
- ////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
- - (NSDateFormatter *)createDateFormatter {
- NSDateFormatter *formatter = [[NSDateFormatter alloc] init];
- [self configureDateFormatter:formatter];
- return formatter;
- }
- - (void)configureDateFormatter:(NSDateFormatter *)dateFormatter {
- [dateFormatter setFormatterBehavior:NSDateFormatterBehavior10_4];
- [dateFormatter setDateFormat:@"yyyy-MM-dd HH:mm:ss:SSS"];
- [dateFormatter setLocale:[NSLocale localeWithLocaleIdentifier:@"en_US_POSIX"]];
- [dateFormatter setCalendar:[[NSCalendar alloc] initWithCalendarIdentifier:NSCalendarIdentifierGregorian]];
- }
- - (NSString *)stringFromDate:(NSDate *)date {
- return [_dateFormatter stringFromDate:date];
- }
- - (NSString *)queueThreadLabelForLogMessage:(DDLogMessage *)logMessage {
- // As per the DDLogFormatter contract, this method is always invoked on the same thread/dispatch_queue
- __auto_type useQueueLabel = NO;
- if (logMessage->_queueLabel) {
- useQueueLabel = YES;
- // If you manually create a thread, it's dispatch_queue will have one of the thread names below.
- // Since all such threads have the same name, we'd prefer to use the threadName or the machThreadID.
- const NSArray<NSString *> *names = @[
- @"com.apple.root.low-priority",
- @"com.apple.root.default-priority",
- @"com.apple.root.high-priority",
- @"com.apple.root.low-overcommit-priority",
- @"com.apple.root.default-overcommit-priority",
- @"com.apple.root.high-overcommit-priority",
- @"com.apple.root.default-qos.overcommit",
- ];
- for (NSString *name in names) {
- if ([logMessage->_queueLabel isEqualToString:name]) {
- useQueueLabel = NO;
- break;
- }
- }
- }
- // Get the name of the queue, thread, or machID (whichever we are to use).
- NSString *queueThreadLabel;
- if (useQueueLabel || [logMessage->_threadName length] > 0) {
- __auto_type fullLabel = useQueueLabel ? logMessage->_queueLabel : logMessage->_threadName;
- NSString *abrvLabel;
- pthread_mutex_lock(&_mutex);
- {
- abrvLabel = _replacements[fullLabel];
- }
- pthread_mutex_unlock(&_mutex);
- queueThreadLabel = abrvLabel ?: fullLabel;
- } else {
- queueThreadLabel = logMessage->_threadID;
- }
- // Now use the thread label in the output
- // labelLength > maxQueueLength : truncate
- // labelLength < minQueueLength : padding
- // : exact
- __auto_type minQueueLength = self.minQueueLength;
- __auto_type maxQueueLength = self.maxQueueLength;
- __auto_type labelLength = [queueThreadLabel length];
- if (maxQueueLength > 0 && labelLength > maxQueueLength) {
- // Truncate
- return [queueThreadLabel substringToIndex:maxQueueLength];
- } else if (labelLength < minQueueLength) {
- // Padding
- return [queueThreadLabel stringByPaddingToLength:minQueueLength
- withString:@" "
- startingAtIndex:0];
- } else {
- // Exact
- return queueThreadLabel;
- }
- }
- - (NSString *)formatLogMessage:(DDLogMessage *)logMessage {
- __auto_type timestamp = [self stringFromDate:logMessage->_timestamp];
- __auto_type queueThreadLabel = [self queueThreadLabelForLogMessage:logMessage];
- return [NSString stringWithFormat:@"%@ [%@ (QOS:%@)] %@", timestamp, queueThreadLabel, _qos_name(logMessage->_qos), logMessage->_message];
- }
- @end
- #pragma mark - DDAtomicCounter
- @interface DDAtomicCounter() {
- atomic_int_fast32_t _value;
- }
- @end
- #pragma clang diagnostic push
- #pragma clang diagnostic ignored "-Wdeprecated-implementations"
- @implementation DDAtomicCounter
- #pragma clang diagnostic pop
- - (instancetype)initWithDefaultValue:(int32_t)defaultValue {
- if ((self = [super init])) {
- atomic_init(&_value, defaultValue);
- }
- return self;
- }
- - (int32_t)value {
- return atomic_load_explicit(&_value, memory_order_relaxed);
- }
- - (int32_t)increment {
- int32_t old = atomic_fetch_add_explicit(&_value, 1, memory_order_relaxed);
- return (old + 1);
- }
- - (int32_t)decrement {
- int32_t old = atomic_fetch_sub_explicit(&_value, 1, memory_order_relaxed);
- return (old - 1);
- }
- @end
|