blob: 128912471c36669ba0930682ef1296ce393799ef [file] [log] [blame]
// Copyright 2010 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "base/message_loop/message_pump.h"
#include "base/check.h"
#include "base/message_loop/message_pump_default.h"
#include "base/message_loop/message_pump_for_io.h"
#include "base/message_loop/message_pump_for_ui.h"
#include "base/notreached.h"
#include "base/task/task_features.h"
#include "base/threading/platform_thread.h"
#include "build/build_config.h"
#if BUILDFLAG(IS_APPLE)
#include "base/message_loop/message_pump_apple.h"
#endif
namespace base {
namespace {
constexpr uint64_t kAlignWakeUpsMask = 1;
constexpr uint64_t kLeewayOffset = 1;
constexpr uint64_t PackAlignWakeUpsAndLeeway(bool align_wake_ups,
TimeDelta leeway) {
return (static_cast<uint64_t>(leeway.InMilliseconds()) << kLeewayOffset) |
(align_wake_ups ? kAlignWakeUpsMask : 0);
}
// This stores the current state of |kAlignWakeUps| and leeway. The last bit
// represents if |kAlignWakeUps| is enabled, and the other bits represent the
// leeway value applied to delayed tasks in milliseconds. An atomic is used here
// because the value is queried from multiple threads.
std::atomic<uint64_t> g_align_wake_ups_and_leeway =
PackAlignWakeUpsAndLeeway(false, kDefaultLeeway);
#if BUILDFLAG(IS_WIN)
bool g_explicit_high_resolution_timer_win = true;
#endif // BUILDFLAG(IS_WIN)
MessagePump::MessagePumpFactory* message_pump_for_ui_factory_ = nullptr;
} // namespace
MessagePump::MessagePump() = default;
MessagePump::~MessagePump() = default;
bool MessagePump::HandleNestedNativeLoopWithApplicationTasks(
bool application_tasks_desired) {
return false;
}
// static
void MessagePump::OverrideMessagePumpForUIFactory(MessagePumpFactory* factory) {
DCHECK(!message_pump_for_ui_factory_);
message_pump_for_ui_factory_ = factory;
}
// static
bool MessagePump::IsMessagePumpForUIFactoryOveridden() {
return message_pump_for_ui_factory_ != nullptr;
}
// static
std::unique_ptr<MessagePump> MessagePump::Create(MessagePumpType type) {
switch (type) {
case MessagePumpType::UI:
if (message_pump_for_ui_factory_)
return message_pump_for_ui_factory_();
#if BUILDFLAG(IS_APPLE)
return message_pump_apple::Create();
#elif BUILDFLAG(IS_NACL) || BUILDFLAG(IS_AIX)
// Currently NaCl and AIX don't have a UI MessagePump.
// TODO(abarth): Figure out if we need this.
NOTREACHED();
return nullptr;
#elif BUILDFLAG(IS_ANDROID)
{
auto message_pump = std::make_unique<MessagePumpAndroid>();
message_pump->set_is_type_ui(true);
return message_pump;
}
#else
return std::make_unique<MessagePumpForUI>();
#endif
case MessagePumpType::IO:
return std::make_unique<MessagePumpForIO>();
#if BUILDFLAG(IS_ANDROID)
case MessagePumpType::JAVA:
return std::make_unique<MessagePumpAndroid>();
#endif
#if BUILDFLAG(IS_APPLE)
case MessagePumpType::NS_RUNLOOP:
return std::make_unique<MessagePumpNSRunLoop>();
#endif
case MessagePumpType::CUSTOM:
NOTREACHED();
return nullptr;
case MessagePumpType::DEFAULT:
#if BUILDFLAG(IS_IOS)
// On iOS, a native runloop is always required to pump system work.
return std::make_unique<MessagePumpCFRunLoop>();
#else
return std::make_unique<MessagePumpDefault>();
#endif
}
}
// static
void MessagePump::InitializeFeatures() {
ResetAlignWakeUpsState();
#if BUILDFLAG(IS_WIN)
g_explicit_high_resolution_timer_win =
FeatureList::IsEnabled(kExplicitHighResolutionTimerWin);
MessagePumpWin::InitializeFeatures();
#endif
}
// static
void MessagePump::OverrideAlignWakeUpsState(bool enabled, TimeDelta leeway) {
g_align_wake_ups_and_leeway.store(PackAlignWakeUpsAndLeeway(enabled, leeway),
std::memory_order_relaxed);
}
// static
void MessagePump::ResetAlignWakeUpsState() {
OverrideAlignWakeUpsState(FeatureList::IsEnabled(kAlignWakeUps),
kTaskLeewayParam.Get());
}
// static
bool MessagePump::GetAlignWakeUpsEnabled() {
return g_align_wake_ups_and_leeway.load(std::memory_order_relaxed) &
kAlignWakeUpsMask;
}
// static
TimeDelta MessagePump::GetLeewayIgnoringThreadOverride() {
return Milliseconds(
g_align_wake_ups_and_leeway.load(std::memory_order_relaxed) >>
kLeewayOffset);
}
// static
TimeDelta MessagePump::GetLeewayForCurrentThread() {
// For some threads, there might be an override of the leeway, so check it
// first.
auto leeway_override = PlatformThread::GetThreadLeewayOverride();
if (leeway_override.has_value()) {
return leeway_override.value();
}
return GetLeewayIgnoringThreadOverride();
}
TimeTicks MessagePump::AdjustDelayedRunTime(TimeTicks earliest_time,
TimeTicks run_time,
TimeTicks latest_time) {
// Windows relies on the low resolution timer rather than manual wake up
// alignment when the leeway is less than the OS default timer resolution.
#if BUILDFLAG(IS_WIN)
if (g_explicit_high_resolution_timer_win &&
GetLeewayForCurrentThread() <=
Milliseconds(Time::kMinLowResolutionThresholdMs)) {
return earliest_time;
}
#endif // BUILDFLAG(IS_WIN)
if (GetAlignWakeUpsEnabled()) {
TimeTicks aligned_run_time = earliest_time.SnappedToNextTick(
TimeTicks(), GetLeewayForCurrentThread());
return std::min(aligned_run_time, latest_time);
}
return run_time;
}
} // namespace base