// Copyright 2015 The Chromium OS 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 <brillo/message_loops/fake_message_loop.h>
#include <base/logging.h>
#include <brillo/location_logging.h>
namespace brillo {
FakeMessageLoop::FakeMessageLoop(base::SimpleTestClock* clock)
: test_clock_(clock) {
}
MessageLoop::TaskId FakeMessageLoop::PostDelayedTask(
const base::Location& from_here,
const base::Closure& task,
base::TimeDelta delay) {
// If no SimpleTestClock was provided, we use the last time we fired a
// callback. In this way, tasks scheduled from a Closure will have the right
// time.
if (test_clock_)
current_time_ = test_clock_->Now();
MessageLoop::TaskId current_id = ++last_id_;
// FakeMessageLoop is limited to only 2^64 tasks. That should be enough.
CHECK(current_id);
tasks_.emplace(current_id, ScheduledTask{from_here, false, task});
fire_order_.push(std::make_pair(current_time_ + delay, current_id));
VLOG_LOC(from_here, 1) << "Scheduling delayed task_id " << current_id
<< " to run at " << current_time_ + delay
<< " (in " << delay << ").";
return current_id;
}
MessageLoop::TaskId FakeMessageLoop::WatchFileDescriptor(
const base::Location& from_here,
int fd,
WatchMode mode,
bool persistent,
const base::Closure& task) {
MessageLoop::TaskId current_id = ++last_id_;
// FakeMessageLoop is limited to only 2^64 tasks. That should be enough.
CHECK(current_id);
tasks_.emplace(current_id, ScheduledTask{from_here, persistent, task});
fds_watched_.emplace(std::make_pair(fd, mode), current_id);
return current_id;
}
bool FakeMessageLoop::CancelTask(TaskId task_id) {
if (task_id == MessageLoop::kTaskIdNull)
return false;
bool ret = tasks_.erase(task_id) > 0;
VLOG_IF(1, ret) << "Removing task_id " << task_id;
return ret;
}
bool FakeMessageLoop::RunOnce(bool may_block) {
if (test_clock_)
current_time_ = test_clock_->Now();
// Try to fire ready file descriptors first.
for (const auto& fd_mode : fds_ready_) {
const auto& fd_watched = fds_watched_.find(fd_mode);
if (fd_watched == fds_watched_.end())
continue;
// The fd_watched->second task might have been canceled and we never removed
// it from the fds_watched_, so we fix that now.
const auto& scheduled_task_ref = tasks_.find(fd_watched->second);
if (scheduled_task_ref == tasks_.end()) {
fds_watched_.erase(fd_watched);
continue;
}
VLOG_LOC(scheduled_task_ref->second.location, 1)
<< "Running task_id " << fd_watched->second
<< " for watching file descriptor " << fd_mode.first << " for "
<< (fd_mode.second == MessageLoop::kWatchRead ? "reading" : "writing")
<< (scheduled_task_ref->second.persistent ?
" persistently" : " just once")
<< " scheduled from this location.";
if (scheduled_task_ref->second.persistent) {
scheduled_task_ref->second.callback.Run();
} else {
base::Closure callback = std::move(scheduled_task_ref->second.callback);
tasks_.erase(scheduled_task_ref);
fds_watched_.erase(fd_watched);
callback.Run();
}
return true;
}
// Try to fire time-based callbacks afterwards.
while (!fire_order_.empty() &&
(may_block || fire_order_.top().first <= current_time_)) {
const auto task_ref = fire_order_.top();
fire_order_.pop();
// We need to skip tasks in the priority_queue not in the |tasks_| map.
// This is normal if the task was canceled, as there is no efficient way
// to remove a task from the priority_queue.
const auto scheduled_task_ref = tasks_.find(task_ref.second);
if (scheduled_task_ref == tasks_.end())
continue;
// Advance the clock to the task firing time, if needed.
if (current_time_ < task_ref.first) {
current_time_ = task_ref.first;
if (test_clock_)
test_clock_->SetNow(current_time_);
}
// Move the Closure out of the map before delete it. We need to delete the
// entry from the map before we call the callback, since calling CancelTask
// for the task you are running now should fail and return false.
base::Closure callback = std::move(scheduled_task_ref->second.callback);
VLOG_LOC(scheduled_task_ref->second.location, 1)
<< "Running task_id " << task_ref.second
<< " at time " << current_time_ << " from this location.";
tasks_.erase(scheduled_task_ref);
callback.Run();
return true;
}
return false;
}
void FakeMessageLoop::SetFileDescriptorReadiness(int fd,
WatchMode mode,
bool ready) {
if (ready)
fds_ready_.emplace(fd, mode);
else
fds_ready_.erase(std::make_pair(fd, mode));
}
bool FakeMessageLoop::PendingTasks() {
for (const auto& task : tasks_) {
VLOG_LOC(task.second.location, 1)
<< "Pending " << (task.second.persistent ? "persistent " : "")
<< "task_id " << task.first << " scheduled from here.";
}
return !tasks_.empty();
}
} // namespace brillo