// Copyright (c) 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "base/ios/scoped_critical_action.h" #import #include #include "base/ios/ios_util.h" #include "base/logging.h" #include "base/memory/ref_counted.h" #include "base/metrics/histogram_macros.h" #include "base/metrics/user_metrics.h" #include "base/strings/sys_string_conversions.h" #include "base/synchronization/lock.h" #import "build/branding_buildflags.h" #if BUILDFLAG(CHROMIUM_BRANDING) #include #endif // BUILDFLAG(CHROMIUM_BRANDING) namespace { // |backgroundTimeRemaining| is thread-safe, but as of Xcode 11.4 this method // is still incorrectly marked as not thread-safe, so checking the value of // |backgroundTimeRemaining| will trigger libMainThreadChecker (if enabled). // Instead, for Chromium builds only, disable libMainThreadChecker for just // this call. This logic is only useful for developer builds, and should not // be included in official builds. These blocks should be removed if future // versions of the library whitelists |backgroundTimeRemaining|. NSTimeInterval GetBackgroundTimeRemaining(UIApplication* application) { #if BUILDFLAG(CHROMIUM_BRANDING) if (!base::ios::IsRunningOnIOS13OrLater()) { // On developer iOS12 builds there's no way to suppress the main thread // checker assert. Since it's a developer build, simply return 0. return 0; } static char const* const lib_main_thread_checker_bundle_path = #if TARGET_IPHONE_SIMULATOR "/usr/lib/libMainThreadChecker.dylib"; #else "/Developer/usr/lib/libMainThreadChecker.dylib"; #endif static void* handle = dlopen(lib_main_thread_checker_bundle_path, RTLD_NOLOAD | RTLD_LAZY); static void (*main_thread_checker_suppression_begin)() = (void (*)())dlsym(handle, "__main_thread_checker_suppression_begin"); if (main_thread_checker_suppression_begin) main_thread_checker_suppression_begin(); #endif // BUILDFLAG(CHROMIUM_BRANDING) NSTimeInterval time = application.backgroundTimeRemaining; #if BUILDFLAG(CHROMIUM_BRANDING) static void (*main_thread_checker_suppression_end)() = (void (*)())dlsym(handle, "__main_thread_checker_suppression_end"); if (main_thread_checker_suppression_end) main_thread_checker_suppression_end(); dlclose(handle); #endif // BUILDFLAG(CHROMIUM_BRANDING) return time; } } // namespace namespace base { namespace ios { ScopedCriticalAction::ScopedCriticalAction(StringPiece task_name) : core_(MakeRefCounted()) { ScopedCriticalAction::Core::StartBackgroundTask(core_, task_name); } ScopedCriticalAction::~ScopedCriticalAction() { ScopedCriticalAction::Core::EndBackgroundTask(core_); } ScopedCriticalAction::Core::Core() : background_task_id_(UIBackgroundTaskInvalid) {} ScopedCriticalAction::Core::~Core() { DCHECK_EQ(background_task_id_, UIBackgroundTaskInvalid); } // This implementation calls |beginBackgroundTaskWithName:expirationHandler:| // when instantiated and |endBackgroundTask:| when destroyed, creating a scope // whose execution will continue (temporarily) even after the app is // backgrounded. // static void ScopedCriticalAction::Core::StartBackgroundTask(scoped_refptr core, StringPiece task_name) { UIApplication* application = [UIApplication sharedApplication]; if (!application) { return; } NSTimeInterval time = GetBackgroundTimeRemaining(application); if (time != DBL_MAX && time > 0) { UMA_HISTOGRAM_MEDIUM_TIMES("IOS.CriticalActionBackgroundTimeRemaining", base::TimeDelta::FromSeconds(time)); } NSString* task_string = !task_name.empty() ? base::SysUTF8ToNSString(task_name) : nil; core->background_task_id_ = [application beginBackgroundTaskWithName:task_string expirationHandler:^{ DLOG(WARNING) << "Background task with name <" << base::SysNSStringToUTF8(task_string) << "> and with " << "id " << core->background_task_id_ << " expired."; // Note if |endBackgroundTask:| is not called for each task // before time expires, the system kills the application. EndBackgroundTask(core); }]; if (core->background_task_id_ == UIBackgroundTaskInvalid) { DLOG(WARNING) << "beginBackgroundTaskWithName:<" << task_name << "> " << "expirationHandler: returned an invalid ID"; } else { VLOG(3) << "Beginning background task <" << task_name << "> with id " << core->background_task_id_; } } // static void ScopedCriticalAction::Core::EndBackgroundTask(scoped_refptr core) { UIBackgroundTaskIdentifier task_id; { AutoLock lock_scope(core->background_task_id_lock_); if (core->background_task_id_ == UIBackgroundTaskInvalid) { return; } task_id = core->background_task_id_; core->background_task_id_ = UIBackgroundTaskInvalid; } VLOG(3) << "Ending background task with id " << task_id; [[UIApplication sharedApplication] endBackgroundTask:task_id]; } } // namespace ios } // namespace base