mirror of
https://github.com/klzgrad/naiveproxy.git
synced 2024-12-05 03:36:08 +03:00
128 lines
4.6 KiB
C
128 lines
4.6 KiB
C
|
// Copyright 2012 The Chromium Authors
|
||
|
// Use of this source code is governed by a BSD-style license that can be
|
||
|
// found in the LICENSE file.
|
||
|
|
||
|
#ifndef BASE_CRITICAL_CLOSURE_H_
|
||
|
#define BASE_CRITICAL_CLOSURE_H_
|
||
|
|
||
|
#include <utility>
|
||
|
|
||
|
#include "base/functional/callback.h"
|
||
|
#include "base/location.h"
|
||
|
#include "base/strings/string_piece.h"
|
||
|
#include "build/build_config.h"
|
||
|
|
||
|
#if BUILDFLAG(IS_IOS)
|
||
|
#include "base/functional/bind.h"
|
||
|
#include "base/ios/scoped_critical_action.h"
|
||
|
#include "third_party/abseil-cpp/absl/types/optional.h"
|
||
|
#endif
|
||
|
|
||
|
namespace base {
|
||
|
|
||
|
namespace internal {
|
||
|
|
||
|
#if BUILDFLAG(IS_IOS)
|
||
|
// This class wraps a closure so it can continue to run for a period of time
|
||
|
// when the application goes to the background by using
|
||
|
// |ios::ScopedCriticalAction|.
|
||
|
class ImmediateCriticalClosure {
|
||
|
public:
|
||
|
explicit ImmediateCriticalClosure(StringPiece task_name, OnceClosure closure);
|
||
|
ImmediateCriticalClosure(const ImmediateCriticalClosure&) = delete;
|
||
|
ImmediateCriticalClosure& operator=(const ImmediateCriticalClosure&) = delete;
|
||
|
~ImmediateCriticalClosure();
|
||
|
void Run();
|
||
|
|
||
|
private:
|
||
|
ios::ScopedCriticalAction critical_action_;
|
||
|
OnceClosure closure_;
|
||
|
};
|
||
|
|
||
|
// This class is identical to ImmediateCriticalClosure, but the critical action
|
||
|
// is started when the action runs, not when the CriticalAction is created.
|
||
|
class PendingCriticalClosure {
|
||
|
public:
|
||
|
explicit PendingCriticalClosure(StringPiece task_name, OnceClosure closure);
|
||
|
PendingCriticalClosure(const PendingCriticalClosure&) = delete;
|
||
|
PendingCriticalClosure& operator=(const PendingCriticalClosure&) = delete;
|
||
|
~PendingCriticalClosure();
|
||
|
void Run();
|
||
|
|
||
|
private:
|
||
|
absl::optional<ios::ScopedCriticalAction> critical_action_;
|
||
|
std::string task_name_;
|
||
|
OnceClosure closure_;
|
||
|
};
|
||
|
#endif // BUILDFLAG(IS_IOS)
|
||
|
|
||
|
} // namespace internal
|
||
|
|
||
|
// Returns a closure that will continue to run for a period of time when the
|
||
|
// application goes to the background if possible on platforms where
|
||
|
// applications don't execute while backgrounded, otherwise the original task is
|
||
|
// returned. If |is_immediate| is true, the closure will immediately prevent
|
||
|
// background suspension. Otherwise, the closure will wait to request background
|
||
|
// permission until it is run.
|
||
|
//
|
||
|
// Example:
|
||
|
// file_task_runner_->PostTask(
|
||
|
// FROM_HERE,
|
||
|
// MakeCriticalClosure(task_name,
|
||
|
// base::BindOnce(&WriteToDiskTask, path_, data)));
|
||
|
//
|
||
|
// Note new closures might be posted in this closure. If the new closures need
|
||
|
// background running time, |MakeCriticalClosure| should be applied on them
|
||
|
// before posting. |task_name| is used by the platform to identify any tasks
|
||
|
// that do not complete in time for suspension.
|
||
|
//
|
||
|
// This function is used automatically for tasks posted to a sequence runner
|
||
|
// using TaskShutdownBehavior::BLOCK_SHUTDOWN.
|
||
|
#if BUILDFLAG(IS_IOS)
|
||
|
inline OnceClosure MakeCriticalClosure(StringPiece task_name,
|
||
|
OnceClosure closure,
|
||
|
bool is_immediate) {
|
||
|
// Wrapping a null closure in a critical closure has unclear semantics and
|
||
|
// most likely indicates a bug. CHECK-ing early allows detecting and
|
||
|
// investigating these cases more easily.
|
||
|
CHECK(!closure.is_null());
|
||
|
if (is_immediate) {
|
||
|
return base::BindOnce(&internal::ImmediateCriticalClosure::Run,
|
||
|
Owned(new internal::ImmediateCriticalClosure(
|
||
|
task_name, std::move(closure))));
|
||
|
} else {
|
||
|
return base::BindOnce(&internal::PendingCriticalClosure::Run,
|
||
|
Owned(new internal::PendingCriticalClosure(
|
||
|
task_name, std::move(closure))));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
inline OnceClosure MakeCriticalClosure(const Location& posted_from,
|
||
|
OnceClosure closure,
|
||
|
bool is_immediate) {
|
||
|
return MakeCriticalClosure(posted_from.ToString(), std::move(closure),
|
||
|
is_immediate);
|
||
|
}
|
||
|
|
||
|
#else // BUILDFLAG(IS_IOS)
|
||
|
|
||
|
inline OnceClosure MakeCriticalClosure(StringPiece task_name,
|
||
|
OnceClosure closure,
|
||
|
bool is_immediate) {
|
||
|
// No-op for platforms where the application does not need to acquire
|
||
|
// background time for closures to finish when it goes into the background.
|
||
|
return closure;
|
||
|
}
|
||
|
|
||
|
inline OnceClosure MakeCriticalClosure(const Location& posted_from,
|
||
|
OnceClosure closure,
|
||
|
bool is_immediate) {
|
||
|
return closure;
|
||
|
}
|
||
|
|
||
|
#endif // BUILDFLAG(IS_IOS)
|
||
|
|
||
|
} // namespace base
|
||
|
|
||
|
#endif // BASE_CRITICAL_CLOSURE_H_
|