|  | // 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 <UIKit/UIKit.h> | 
|  |  | 
|  | #include "base/logging.h" | 
|  | #include "base/memory/ref_counted.h" | 
|  | #include "base/synchronization/lock.h" | 
|  |  | 
|  | namespace base { | 
|  | namespace ios { | 
|  |  | 
|  | ScopedCriticalAction::ScopedCriticalAction() | 
|  | : core_(MakeRefCounted<ScopedCriticalAction::Core>()) { | 
|  | ScopedCriticalAction::Core::StartBackgroundTask(core_); | 
|  | } | 
|  |  | 
|  | 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 |beginBackgroundTaskWithExpirationHandler:| 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> core) { | 
|  | UIApplication* application = [UIApplication sharedApplication]; | 
|  | if (!application) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | core->background_task_id_ = | 
|  | [application beginBackgroundTaskWithExpirationHandler:^{ | 
|  | DLOG(WARNING) << "Background task 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) | 
|  | << "beginBackgroundTaskWithExpirationHandler: returned an invalid ID"; | 
|  | } else { | 
|  | VLOG(3) << "Beginning background task with id " | 
|  | << core->background_task_id_; | 
|  | } | 
|  | } | 
|  |  | 
|  | // static | 
|  | void ScopedCriticalAction::Core::EndBackgroundTask(scoped_refptr<Core> 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 |