425 строки
13 KiB
C
425 строки
13 KiB
C
/* -*- mode: C; c-basic-offset: 4; indent-tabs-mode: nil; -*- */
|
|
/* vim:set et sts=4: */
|
|
/* bus - The Input Bus
|
|
* Copyright (C) 2010 Peng Huang <shawn.p.huang@gmail.com>
|
|
* Copyright (C) 2010 Google Inc.
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301
|
|
* USA
|
|
*/
|
|
#include "component.h"
|
|
|
|
#include <gio/gio.h>
|
|
#include <glib/gstdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#include "global.h"
|
|
#include "marshalers.h"
|
|
#include "types.h"
|
|
|
|
enum {
|
|
LAST_SIGNAL,
|
|
};
|
|
|
|
enum {
|
|
PROP_0 = 0,
|
|
PROP_COMPONENT,
|
|
PROP_FACTORY,
|
|
};
|
|
|
|
struct _BusComponent {
|
|
IBusObject parent;
|
|
|
|
/* instance members */
|
|
|
|
/* an object which represents one XML file in the ibus/component/ directory. */
|
|
IBusComponent *component;
|
|
/* a proxy object which starts an engine. */
|
|
BusFactoryProxy *factory;
|
|
|
|
/* TRUE if the component started in the verbose mode. */
|
|
gboolean verbose;
|
|
/* TRUE if the component needs to be restarted when it dies. */
|
|
gboolean restart;
|
|
/* TRUE if the component will be destroyed with factory. */
|
|
gboolean destroy_with_factory;
|
|
|
|
/* process id of the process (e.g. ibus-config, ibus-engine-*, ..) of the component. */
|
|
GPid pid;
|
|
guint child_source_id;
|
|
};
|
|
|
|
struct _BusComponentClass {
|
|
IBusObjectClass parent;
|
|
/* class members */
|
|
};
|
|
|
|
/* functions prototype */
|
|
static GObject* bus_component_constructor (GType type,
|
|
guint n_construct_params,
|
|
GObjectConstructParam *construct_params);
|
|
static void bus_component_set_property (BusComponent *component,
|
|
guint prop_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec);
|
|
static void bus_component_get_property (BusComponent *component,
|
|
guint prop_id,
|
|
GValue *value,
|
|
GParamSpec *pspec);
|
|
static void bus_component_destroy (BusComponent *component);
|
|
|
|
G_DEFINE_TYPE (BusComponent, bus_component, IBUS_TYPE_OBJECT)
|
|
|
|
static void
|
|
bus_component_class_init (BusComponentClass *class)
|
|
{
|
|
GObjectClass *gobject_class = G_OBJECT_CLASS (class);
|
|
IBusObjectClass *ibus_object_class = IBUS_OBJECT_CLASS (class);
|
|
|
|
gobject_class->constructor = bus_component_constructor;
|
|
gobject_class->set_property = (GObjectSetPropertyFunc) bus_component_set_property;
|
|
gobject_class->get_property = (GObjectGetPropertyFunc) bus_component_get_property;
|
|
ibus_object_class->destroy = (IBusObjectDestroyFunc) bus_component_destroy;
|
|
|
|
/* install properties */
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_COMPONENT,
|
|
g_param_spec_object ("component", /* canonical name of the property */
|
|
"component", /* nick name */
|
|
"component", /* description */
|
|
IBUS_TYPE_COMPONENT, /* object type */
|
|
G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY));
|
|
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_FACTORY,
|
|
g_param_spec_object ("factory",
|
|
"factory",
|
|
"factory",
|
|
BUS_TYPE_FACTORY_PROXY,
|
|
G_PARAM_READWRITE));
|
|
}
|
|
|
|
static void
|
|
bus_component_init (BusComponent *component)
|
|
{
|
|
}
|
|
|
|
/**
|
|
* bus_component_constructor:
|
|
*
|
|
* A constructor method which is called after bus_component_init is called.
|
|
*/
|
|
static GObject*
|
|
bus_component_constructor (GType type,
|
|
guint n_construct_params,
|
|
GObjectConstructParam *construct_params)
|
|
{
|
|
GObject *object;
|
|
object = G_OBJECT_CLASS (bus_component_parent_class)->constructor (type,
|
|
n_construct_params,
|
|
construct_params);
|
|
BusComponent *component = (BusComponent *) object;
|
|
/* we have to override the _constructor method since in _init method, the component->component property is not set yet. */
|
|
g_assert (IBUS_IS_COMPONENT (component->component));
|
|
|
|
static GQuark quark = 0;
|
|
if (quark == 0) {
|
|
quark = g_quark_from_static_string ("BusComponent");
|
|
}
|
|
|
|
/* associate each engine with BusComponent. a component might have one or more components. For example, ibus-engine-pinyin would
|
|
* have two - 'pinyin' and 'bopomofo' and ibus-engine-m17n has many. On the other hand, the gtkpanel component does not have an
|
|
* engine, of course. */
|
|
GList *engines = ibus_component_get_engines (component->component);
|
|
GList *p;
|
|
for (p = engines; p != NULL; p = p->next) {
|
|
g_object_set_qdata ((GObject *) p->data, quark, component);
|
|
}
|
|
g_list_free (engines);
|
|
|
|
return object;
|
|
}
|
|
|
|
static void
|
|
bus_component_set_property (BusComponent *component,
|
|
guint prop_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
switch (prop_id) {
|
|
case PROP_COMPONENT:
|
|
g_assert (component->component == NULL);
|
|
component->component = g_value_dup_object (value);
|
|
break;
|
|
case PROP_FACTORY:
|
|
bus_component_set_factory (component, (BusFactoryProxy *) g_value_get_object (value));
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (component, prop_id, pspec);
|
|
}
|
|
}
|
|
|
|
static void
|
|
bus_component_get_property (BusComponent *component,
|
|
guint prop_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
switch (prop_id) {
|
|
case PROP_COMPONENT:
|
|
g_value_set_object (value, bus_component_get_component (component));
|
|
break;
|
|
case PROP_FACTORY:
|
|
g_value_set_object (value, bus_component_get_factory (component));
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (component, prop_id, pspec);
|
|
}
|
|
}
|
|
|
|
static void
|
|
bus_component_destroy (BusComponent *component)
|
|
{
|
|
if (component->pid != 0) {
|
|
bus_component_stop (component);
|
|
g_spawn_close_pid (component->pid);
|
|
component->pid = 0;
|
|
}
|
|
|
|
if (component->child_source_id != 0) {
|
|
g_source_remove (component->child_source_id);
|
|
component->child_source_id = 0;
|
|
}
|
|
|
|
if (component->component != NULL) {
|
|
g_object_unref (component->component);
|
|
component->component = NULL;
|
|
}
|
|
|
|
IBUS_OBJECT_CLASS (bus_component_parent_class)->destroy (IBUS_OBJECT (component));
|
|
}
|
|
|
|
BusComponent *
|
|
bus_component_new (IBusComponent *component,
|
|
BusFactoryProxy *factory)
|
|
{
|
|
g_assert (IBUS_IS_COMPONENT (component));
|
|
|
|
return (BusComponent *) g_object_new (BUS_TYPE_COMPONENT,
|
|
/* properties below will be set via the bus_component_set_property function. */
|
|
"component", component,
|
|
"factory", factory,
|
|
NULL);
|
|
}
|
|
|
|
static void
|
|
bus_component_factory_destroy_cb (BusFactoryProxy *factory,
|
|
BusComponent *component)
|
|
{
|
|
g_return_if_fail (component->factory == factory);
|
|
|
|
g_object_unref (component->factory);
|
|
component->factory = NULL;
|
|
/* emit the "notify" signal for the factory property on component. */
|
|
g_object_notify ((GObject *) component, "factory");
|
|
|
|
if (component->destroy_with_factory)
|
|
ibus_object_destroy ((IBusObject *) component);
|
|
}
|
|
|
|
IBusComponent *
|
|
bus_component_get_component (BusComponent *component)
|
|
{
|
|
g_assert (BUS_IS_COMPONENT (component));
|
|
return component->component;
|
|
}
|
|
|
|
void
|
|
bus_component_set_factory (BusComponent *component,
|
|
BusFactoryProxy *factory)
|
|
{
|
|
g_assert (BUS_IS_COMPONENT (component));
|
|
|
|
if (component->factory == factory) {
|
|
return;
|
|
}
|
|
|
|
if (component->factory) {
|
|
g_signal_handlers_disconnect_by_func (component->factory,
|
|
bus_component_factory_destroy_cb,
|
|
component);
|
|
g_object_unref (component->factory);
|
|
component->factory = NULL;
|
|
}
|
|
|
|
if (factory) {
|
|
g_assert (BUS_IS_FACTORY_PROXY (factory));
|
|
component->factory = (BusFactoryProxy *) g_object_ref (factory);
|
|
g_signal_connect (factory, "destroy",
|
|
G_CALLBACK (bus_component_factory_destroy_cb), component);
|
|
}
|
|
|
|
/* emit the "notify" signal for the factory property on component. */
|
|
g_object_notify ((GObject*) component, "factory");
|
|
}
|
|
|
|
BusFactoryProxy *
|
|
bus_component_get_factory (BusComponent *component)
|
|
{
|
|
g_assert (BUS_IS_COMPONENT (component));
|
|
return component->factory;
|
|
}
|
|
|
|
const gchar *
|
|
bus_component_get_name (BusComponent *component)
|
|
{
|
|
g_assert (BUS_IS_COMPONENT (component));
|
|
|
|
return ibus_component_get_name (component->component);
|
|
}
|
|
|
|
GList *
|
|
bus_component_get_engines (BusComponent *component)
|
|
{
|
|
g_assert (BUS_IS_COMPONENT (component));
|
|
|
|
return ibus_component_get_engines (component->component);
|
|
}
|
|
|
|
void
|
|
bus_component_set_destroy_with_factory (BusComponent *component,
|
|
gboolean with_factory)
|
|
{
|
|
g_assert (BUS_IS_COMPONENT (component));
|
|
|
|
component->destroy_with_factory = with_factory;
|
|
}
|
|
|
|
void
|
|
bus_component_set_restart (BusComponent *component,
|
|
gboolean restart)
|
|
{
|
|
g_assert (BUS_IS_COMPONENT (component));
|
|
component->restart = restart;
|
|
}
|
|
|
|
/**
|
|
* bus_component_child_cb:
|
|
*
|
|
* A callback function to be called when the child process is terminated.
|
|
*/
|
|
static void
|
|
bus_component_child_cb (GPid pid,
|
|
gint status,
|
|
BusComponent *component)
|
|
{
|
|
g_assert (BUS_IS_COMPONENT (component));
|
|
g_assert (component->pid == pid);
|
|
g_spawn_close_pid (pid);
|
|
component->pid = 0;
|
|
component->child_source_id = 0;
|
|
|
|
if (component->restart) {
|
|
bus_component_start (component, component->verbose);
|
|
}
|
|
}
|
|
|
|
gboolean
|
|
bus_component_start (BusComponent *component,
|
|
gboolean verbose)
|
|
{
|
|
g_assert (BUS_IS_COMPONENT (component));
|
|
|
|
if (component->pid != 0)
|
|
return TRUE;
|
|
|
|
component->verbose = verbose;
|
|
|
|
gint argc;
|
|
gchar **argv;
|
|
gboolean retval;
|
|
|
|
GError *error = NULL;
|
|
if (!g_shell_parse_argv (ibus_component_get_exec (component->component),
|
|
&argc,
|
|
&argv,
|
|
&error)) {
|
|
g_warning ("Can not parse component %s exec: %s",
|
|
ibus_component_get_name (component->component),
|
|
error->message);
|
|
g_error_free (error);
|
|
return FALSE;
|
|
}
|
|
|
|
error = NULL;
|
|
GSpawnFlags flags = G_SPAWN_DO_NOT_REAP_CHILD;
|
|
if (!verbose) {
|
|
flags |= G_SPAWN_STDOUT_TO_DEV_NULL | G_SPAWN_STDERR_TO_DEV_NULL;
|
|
}
|
|
retval = g_spawn_async (NULL, argv, NULL,
|
|
flags,
|
|
NULL, NULL,
|
|
&(component->pid), &error);
|
|
g_strfreev (argv);
|
|
if (!retval) {
|
|
g_warning ("Can not execute component %s: %s",
|
|
ibus_component_get_name (component->component),
|
|
error->message);
|
|
g_error_free (error);
|
|
return FALSE;
|
|
}
|
|
|
|
component->child_source_id =
|
|
g_child_watch_add (component->pid,
|
|
(GChildWatchFunc) bus_component_child_cb,
|
|
component);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
gboolean
|
|
bus_component_stop (BusComponent *component)
|
|
{
|
|
g_assert (BUS_IS_COMPONENT (component));
|
|
|
|
if (component->pid == 0)
|
|
return TRUE;
|
|
|
|
kill (component->pid, SIGTERM);
|
|
return TRUE;
|
|
}
|
|
|
|
gboolean
|
|
bus_component_is_running (BusComponent *component)
|
|
{
|
|
g_assert (BUS_IS_COMPONENT (component));
|
|
|
|
return (component->pid != 0);
|
|
}
|
|
|
|
BusComponent *
|
|
bus_component_from_engine_desc (IBusEngineDesc *engine)
|
|
{
|
|
g_assert (IBUS_IS_ENGINE_DESC (engine));
|
|
|
|
static GQuark quark = 0;
|
|
if (quark == 0) {
|
|
quark = g_quark_from_static_string ("BusComponent");
|
|
}
|
|
|
|
return (BusComponent *) g_object_get_qdata ((GObject *) engine, quark);
|
|
}
|