// 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/glib_message_loop.h>
#include <fcntl.h>
#include <unistd.h>
#include <brillo/location_logging.h>
using base::Closure;
namespace brillo {
GlibMessageLoop::GlibMessageLoop() {
loop_ = g_main_loop_new(g_main_context_default(), FALSE);
}
GlibMessageLoop::~GlibMessageLoop() {
// Cancel all pending tasks when destroying the message loop.
for (const auto& task : tasks_) {
DVLOG_LOC(task.second->location, 1)
<< "Removing task_id " << task.second->task_id
<< " leaked on GlibMessageLoop, scheduled from this location.";
g_source_remove(task.second->source_id);
}
g_main_loop_unref(loop_);
}
MessageLoop::TaskId GlibMessageLoop::PostDelayedTask(
const tracked_objects::Location& from_here,
const Closure &task,
base::TimeDelta delay) {
TaskId task_id = NextTaskId();
// Note: While we store persistent = false in the ScheduledTask object, we
// don't check it in OnRanPostedTask() since it is always false for delayed
// tasks. This is only used for WatchFileDescriptor below.
ScheduledTask* scheduled_task = new ScheduledTask{
this, from_here, task_id, 0, false, std::move(task)};
DVLOG_LOC(from_here, 1) << "Scheduling delayed task_id " << task_id
<< " to run in " << delay << ".";
scheduled_task->source_id = g_timeout_add_full(
G_PRIORITY_DEFAULT,
delay.InMillisecondsRoundedUp(),
&GlibMessageLoop::OnRanPostedTask,
reinterpret_cast<gpointer>(scheduled_task),
DestroyPostedTask);
tasks_[task_id] = scheduled_task;
return task_id;
}
MessageLoop::TaskId GlibMessageLoop::WatchFileDescriptor(
const tracked_objects::Location& from_here,
int fd,
WatchMode mode,
bool persistent,
const Closure &task) {
// Quick check to see if the fd is valid.
if (fcntl(fd, F_GETFD) == -1 && errno == EBADF)
return MessageLoop::kTaskIdNull;
GIOCondition condition = G_IO_NVAL;
switch (mode) {
case MessageLoop::kWatchRead:
condition = static_cast<GIOCondition>(G_IO_IN | G_IO_HUP | G_IO_NVAL);
break;
case MessageLoop::kWatchWrite:
condition = static_cast<GIOCondition>(G_IO_OUT | G_IO_HUP | G_IO_NVAL);
break;
default:
return MessageLoop::kTaskIdNull;
}
// TODO(deymo): Used g_unix_fd_add_full() instead of g_io_add_watch_full()
// when/if we switch to glib 2.36 or newer so we don't need to create a
// GIOChannel for this.
GIOChannel* io_channel = g_io_channel_unix_new(fd);
if (!io_channel)
return MessageLoop::kTaskIdNull;
GError* error = nullptr;
GIOStatus status = g_io_channel_set_encoding(io_channel, nullptr, &error);
if (status != G_IO_STATUS_NORMAL) {
LOG(ERROR) << "GError(" << error->code << "): "
<< (error->message ? error->message : "(unknown)");
g_error_free(error);
// g_io_channel_set_encoding() documentation states that this should be
// valid in this context (a new io_channel), but enforce the check in
// debug mode.
DCHECK(status == G_IO_STATUS_NORMAL);
return MessageLoop::kTaskIdNull;
}
TaskId task_id = NextTaskId();
ScheduledTask* scheduled_task = new ScheduledTask{
this, from_here, task_id, 0, persistent, std::move(task)};
scheduled_task->source_id = g_io_add_watch_full(
io_channel,
G_PRIORITY_DEFAULT,
condition,
&GlibMessageLoop::OnWatchedFdReady,
reinterpret_cast<gpointer>(scheduled_task),
DestroyPostedTask);
// g_io_add_watch_full() increases the reference count on the newly created
// io_channel, so we can dereference it now and it will be free'd once the
// source is removed or now if g_io_add_watch_full() failed.
g_io_channel_unref(io_channel);
DVLOG_LOC(from_here, 1)
<< "Watching fd " << fd << " for "
<< (mode == MessageLoop::kWatchRead ? "reading" : "writing")
<< (persistent ? " persistently" : " just once")
<< " as task_id " << task_id
<< (scheduled_task->source_id ? " successfully" : " failed.");
if (!scheduled_task->source_id) {
delete scheduled_task;
return MessageLoop::kTaskIdNull;
}
tasks_[task_id] = scheduled_task;
return task_id;
}
bool GlibMessageLoop::CancelTask(TaskId task_id) {
if (task_id == kTaskIdNull)
return false;
const auto task = tasks_.find(task_id);
// It is a programmer error to attempt to remove a non-existent source.
if (task == tasks_.end())
return false;
DVLOG_LOC(task->second->location, 1)
<< "Removing task_id " << task_id << " scheduled from this location.";
guint source_id = task->second->source_id;
// We remove here the entry from the tasks_ map, the pointer will be deleted
// by the g_source_remove() call.
tasks_.erase(task);
return g_source_remove(source_id);
}
bool GlibMessageLoop::RunOnce(bool may_block) {
return g_main_context_iteration(nullptr, may_block);
}
void GlibMessageLoop::Run() {
g_main_loop_run(loop_);
}
void GlibMessageLoop::BreakLoop() {
g_main_loop_quit(loop_);
}
MessageLoop::TaskId GlibMessageLoop::NextTaskId() {
TaskId res;
do {
res = ++last_id_;
// We would run out of memory before we run out of task ids.
} while (!res || tasks_.find(res) != tasks_.end());
return res;
}
gboolean GlibMessageLoop::OnRanPostedTask(gpointer user_data) {
ScheduledTask* scheduled_task = reinterpret_cast<ScheduledTask*>(user_data);
DVLOG_LOC(scheduled_task->location, 1)
<< "Running delayed task_id " << scheduled_task->task_id
<< " scheduled from this location.";
// We only need to remove this task_id from the map. DestroyPostedTask will be
// called with this same |user_data| where we can delete the ScheduledTask.
scheduled_task->loop->tasks_.erase(scheduled_task->task_id);
scheduled_task->closure.Run();
return FALSE; // Removes the source since a callback can only be called once.
}
gboolean GlibMessageLoop::OnWatchedFdReady(GIOChannel *source,
GIOCondition condition,
gpointer user_data) {
ScheduledTask* scheduled_task = reinterpret_cast<ScheduledTask*>(user_data);
DVLOG_LOC(scheduled_task->location, 1)
<< "Running task_id " << scheduled_task->task_id
<< " for watching a file descriptor, scheduled from this location.";
if (!scheduled_task->persistent) {
// We only need to remove this task_id from the map. DestroyPostedTask will
// be called with this same |user_data| where we can delete the
// ScheduledTask.
scheduled_task->loop->tasks_.erase(scheduled_task->task_id);
}
scheduled_task->closure.Run();
return scheduled_task->persistent;
}
void GlibMessageLoop::DestroyPostedTask(gpointer user_data) {
delete reinterpret_cast<ScheduledTask*>(user_data);
}
} // namespace brillo