2005-06-22 18:20:32 +00:00
|
|
|
/*
|
|
|
|
* mooapp/mooappinput.c
|
|
|
|
*
|
2007-04-07 03:21:52 -05:00
|
|
|
* Copyright (C) 2004-2007 by Yevgen Muntyan <muntyan@math.tamu.edu>
|
2005-06-22 18:20:32 +00:00
|
|
|
*
|
2007-06-24 12:56:20 -05:00
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
2007-09-23 11:47:28 -05:00
|
|
|
* License version 2.1 as published by the Free Software Foundation.
|
2005-06-22 18:20:32 +00:00
|
|
|
*
|
|
|
|
* See COPYING file that comes with this distribution.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef __WIN32__
|
2007-07-10 10:32:29 -05:00
|
|
|
#define MOO_APP_INPUT_WIN32
|
|
|
|
#elif defined(MOO_USE_PIPE_INPUT)
|
|
|
|
#define MOO_APP_INPUT_PIPE
|
|
|
|
#else
|
|
|
|
#define MOO_APP_INPUT_SOCKET
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(MOO_APP_INPUT_WIN32)
|
|
|
|
# include <windows.h>
|
|
|
|
# include <io.h>
|
|
|
|
#elif defined(MOO_APP_INPUT_PIPE)
|
|
|
|
# include <sys/poll.h>
|
|
|
|
# include <signal.h>
|
|
|
|
#else
|
|
|
|
# include <sys/socket.h>
|
|
|
|
# include <sys/un.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef __WIN32__
|
2005-06-22 18:20:32 +00:00
|
|
|
#include <unistd.h>
|
|
|
|
#include <stdlib.h>
|
2007-07-10 10:32:29 -05:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#endif
|
2005-06-22 18:20:32 +00:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
#include <fcntl.h>
|
2005-10-13 14:08:18 +00:00
|
|
|
#include <errno.h>
|
2005-06-22 18:20:32 +00:00
|
|
|
#include <stdio.h>
|
2005-11-02 02:32:07 +00:00
|
|
|
#include <string.h>
|
2005-06-22 18:20:32 +00:00
|
|
|
#include "mooapp/mooappinput.h"
|
2006-08-15 02:12:41 -05:00
|
|
|
#define MOO_APP_COMPILATION
|
|
|
|
#include "mooapp/mooapp-private.h"
|
2006-11-02 00:38:00 -06:00
|
|
|
#include "mooutils/mooutils-misc.h"
|
2006-12-19 02:12:37 -06:00
|
|
|
#include "mooutils/mooutils-thread.h"
|
2007-08-20 22:15:58 -05:00
|
|
|
#include "mooutils/mooutils-debug.h"
|
2005-06-22 18:20:32 +00:00
|
|
|
|
2007-08-20 22:15:58 -05:00
|
|
|
MOO_DEBUG_INIT(input, TRUE)
|
2005-06-22 18:20:32 +00:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
#define MAX_BUFFER_SIZE 4096
|
|
|
|
|
|
|
|
#ifdef MOO_APP_INPUT_SOCKET
|
|
|
|
#define INPUT_PREFIX "in-"
|
|
|
|
#else
|
|
|
|
#define INPUT_PREFIX "input-"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
typedef struct InputChannel InputChannel;
|
|
|
|
|
2006-06-03 02:26:18 -05:00
|
|
|
struct _MooAppInput
|
|
|
|
{
|
2007-07-10 10:32:29 -05:00
|
|
|
GSList *pipes;
|
|
|
|
char *appname;
|
2007-07-13 03:42:31 -05:00
|
|
|
char *main_path;
|
2006-06-03 02:26:18 -05:00
|
|
|
};
|
2005-06-22 18:20:32 +00:00
|
|
|
|
2007-07-13 03:42:31 -05:00
|
|
|
static InputChannel *input_channel_new (const char *appname,
|
|
|
|
const char *name,
|
|
|
|
gboolean may_fail);
|
|
|
|
static void input_channel_free (InputChannel *ch);
|
|
|
|
static char *input_channel_get_path (InputChannel *ch);
|
2006-06-03 02:26:18 -05:00
|
|
|
|
2005-06-22 18:20:32 +00:00
|
|
|
|
2006-06-03 02:26:18 -05:00
|
|
|
MooAppInput *
|
2007-07-10 10:32:29 -05:00
|
|
|
_moo_app_input_new (const char *appname,
|
|
|
|
const char *name,
|
|
|
|
gboolean bind_default)
|
2005-06-22 18:20:32 +00:00
|
|
|
{
|
|
|
|
MooAppInput *ch;
|
2007-07-10 10:32:29 -05:00
|
|
|
InputChannel *ich;
|
2005-06-22 18:20:32 +00:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
g_return_val_if_fail (appname != NULL, NULL);
|
2005-06-22 18:20:32 +00:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
ch = _moo_new0 (MooAppInput);
|
2005-06-22 18:20:32 +00:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
ch->pipes = NULL;
|
|
|
|
ch->appname = g_strdup (appname);
|
2005-06-22 18:20:32 +00:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
if ((ich = input_channel_new (appname, _moo_get_pid_string (), FALSE)))
|
2007-07-13 03:42:31 -05:00
|
|
|
{
|
2007-07-10 10:32:29 -05:00
|
|
|
ch->pipes = g_slist_prepend (ch->pipes, ich);
|
2007-07-13 03:42:31 -05:00
|
|
|
ch->main_path = input_channel_get_path (ich);
|
|
|
|
}
|
2005-06-22 18:20:32 +00:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
if (name && (ich = input_channel_new (appname, name, FALSE)))
|
|
|
|
ch->pipes = g_slist_prepend (ch->pipes, ich);
|
2005-06-22 18:20:32 +00:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
if (bind_default && (ich = input_channel_new (appname, MOO_APP_INPUT_NAME_DEFAULT, TRUE)))
|
|
|
|
ch->pipes = g_slist_prepend (ch->pipes, ich);
|
2005-06-22 18:20:32 +00:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
return ch;
|
2005-06-22 18:20:32 +00:00
|
|
|
}
|
|
|
|
|
2005-10-13 14:08:18 +00:00
|
|
|
void
|
2007-07-10 10:32:29 -05:00
|
|
|
_moo_app_input_free (MooAppInput *ch)
|
2005-06-22 18:20:32 +00:00
|
|
|
{
|
|
|
|
g_return_if_fail (ch != NULL);
|
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
g_slist_foreach (ch->pipes, (GFunc) input_channel_free, NULL);
|
2007-08-06 13:17:25 -05:00
|
|
|
g_slist_free (ch->pipes);
|
2005-06-22 18:20:32 +00:00
|
|
|
|
2007-07-13 03:42:31 -05:00
|
|
|
g_free (ch->main_path);
|
2007-07-10 10:32:29 -05:00
|
|
|
g_free (ch->appname);
|
|
|
|
_moo_free (MooAppInput, ch);
|
2006-06-03 02:26:18 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-07-13 03:42:31 -05:00
|
|
|
const char *
|
|
|
|
_moo_app_input_get_path (MooAppInput *ch)
|
|
|
|
{
|
|
|
|
g_return_val_if_fail (ch != NULL, NULL);
|
|
|
|
return ch->main_path;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-10-13 14:08:18 +00:00
|
|
|
static void
|
2007-07-10 10:32:29 -05:00
|
|
|
commit (GString **buffer)
|
2005-06-22 18:20:32 +00:00
|
|
|
{
|
2007-05-14 20:08:25 -05:00
|
|
|
char buf[MAX_BUFFER_SIZE];
|
|
|
|
GString *freeme = NULL;
|
|
|
|
char *ptr;
|
|
|
|
gsize len;
|
2005-06-22 18:20:32 +00:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
if (!(*buffer)->len)
|
2007-05-14 20:08:25 -05:00
|
|
|
{
|
2007-08-20 22:15:58 -05:00
|
|
|
moo_dmsg ("%s: got empty command", G_STRLOC);
|
2007-05-14 20:08:25 -05:00
|
|
|
return;
|
|
|
|
}
|
2005-10-13 14:08:18 +00:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
if ((*buffer)->len + 1 > MAX_BUFFER_SIZE)
|
2005-10-13 14:08:18 +00:00
|
|
|
{
|
2007-07-10 10:32:29 -05:00
|
|
|
freeme = *buffer;
|
|
|
|
*buffer = g_string_new_len (NULL, MAX_BUFFER_SIZE);
|
2007-05-14 20:08:25 -05:00
|
|
|
ptr = freeme->str;
|
|
|
|
len = freeme->len;
|
2005-10-13 14:08:18 +00:00
|
|
|
}
|
2005-06-22 18:20:32 +00:00
|
|
|
else
|
2005-10-13 14:08:18 +00:00
|
|
|
{
|
2007-07-10 10:32:29 -05:00
|
|
|
memcpy (buf, (*buffer)->str, (*buffer)->len + 1);
|
2007-05-14 20:08:25 -05:00
|
|
|
ptr = buf;
|
2007-07-10 10:32:29 -05:00
|
|
|
len = (*buffer)->len;
|
|
|
|
g_string_truncate (*buffer, 0);
|
2005-10-13 14:08:18 +00:00
|
|
|
}
|
|
|
|
|
2007-08-18 02:23:59 -05:00
|
|
|
if (0)
|
2007-05-14 20:08:25 -05:00
|
|
|
g_print ("%s: commit %c\n%s\n-----\n", G_STRLOC, ptr[0], ptr + 1);
|
|
|
|
|
|
|
|
_moo_app_exec_cmd (moo_app_get_instance (), ptr[0], ptr + 1, len - 1);
|
|
|
|
|
|
|
|
if (freeme)
|
|
|
|
g_string_free (freeme, TRUE);
|
2005-06-22 18:20:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
#ifndef MOO_APP_INPUT_WIN32
|
|
|
|
|
|
|
|
static char *
|
|
|
|
get_pipe_dir (const char *appname)
|
|
|
|
{
|
|
|
|
GdkDisplay *display;
|
|
|
|
char *display_name;
|
|
|
|
char *user_name;
|
|
|
|
char *name;
|
2005-06-22 18:20:32 +00:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
g_return_val_if_fail (appname != NULL, NULL);
|
2005-06-22 18:20:32 +00:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
display = gdk_display_get_default ();
|
|
|
|
g_return_val_if_fail (display != NULL, NULL);
|
2005-06-22 18:20:32 +00:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
display_name = g_strcanon (g_strdup (gdk_display_get_name (display)),
|
|
|
|
G_CSET_A_2_Z G_CSET_a_2_z G_CSET_DIGITS,
|
|
|
|
'-');
|
|
|
|
user_name = g_strcanon (g_strdup (g_get_user_name ()),
|
|
|
|
G_CSET_A_2_Z G_CSET_a_2_z G_CSET_DIGITS,
|
|
|
|
'-');
|
|
|
|
|
|
|
|
name = g_strdup_printf ("%s/%s-%s-%s", g_get_tmp_dir (), appname, user_name,
|
|
|
|
display_name[0] == '-' ? &display_name[1] : display_name);
|
|
|
|
|
|
|
|
g_free (display_name);
|
|
|
|
g_free (user_name);
|
|
|
|
return name;
|
|
|
|
}
|
|
|
|
|
|
|
|
static char *
|
|
|
|
get_pipe_path (const char *pipe_dir,
|
|
|
|
const char *name)
|
2005-06-22 18:20:32 +00:00
|
|
|
{
|
2007-07-10 10:32:29 -05:00
|
|
|
return g_strdup_printf ("%s/" INPUT_PREFIX "%s",
|
|
|
|
pipe_dir, name);
|
2005-06-22 18:20:32 +00:00
|
|
|
}
|
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
static gboolean
|
|
|
|
input_channel_start_io (int fd,
|
|
|
|
GIOFunc io_func,
|
|
|
|
gpointer data,
|
|
|
|
GIOChannel **io_channel,
|
|
|
|
guint *io_watch)
|
2005-06-22 18:20:32 +00:00
|
|
|
{
|
2007-07-10 10:32:29 -05:00
|
|
|
GSource *source;
|
|
|
|
|
|
|
|
*io_channel = g_io_channel_unix_new (fd);
|
|
|
|
g_io_channel_set_encoding (*io_channel, NULL, NULL);
|
|
|
|
|
|
|
|
*io_watch = _moo_io_add_watch (*io_channel,
|
|
|
|
G_IO_IN | G_IO_PRI | G_IO_HUP | G_IO_ERR,
|
|
|
|
io_func, data);
|
|
|
|
|
|
|
|
source = g_main_context_find_source_by_id (NULL, *io_watch);
|
|
|
|
g_source_set_can_recurse (source, TRUE);
|
|
|
|
|
|
|
|
return TRUE;
|
2005-06-22 18:20:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
static gboolean do_send (const char *filename,
|
|
|
|
const char *data,
|
|
|
|
gssize data_len);
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
try_send (const char *pipe_dir_name,
|
|
|
|
const char *name,
|
|
|
|
const char *data,
|
|
|
|
gssize data_len)
|
2005-06-22 18:20:32 +00:00
|
|
|
{
|
2007-07-10 10:32:29 -05:00
|
|
|
char *filename = NULL;
|
|
|
|
gboolean result = FALSE;
|
2005-06-22 18:20:32 +00:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
g_return_val_if_fail (name && name[0], FALSE);
|
2005-06-22 18:20:32 +00:00
|
|
|
|
2007-07-31 08:02:07 -05:00
|
|
|
filename = get_pipe_path (pipe_dir_name, name);
|
2007-08-20 22:15:58 -05:00
|
|
|
moo_dmsg ("try_send: sending data to `%s'", filename);
|
2005-06-22 18:20:32 +00:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
if (!g_file_test (filename, G_FILE_TEST_EXISTS))
|
2005-06-22 18:20:32 +00:00
|
|
|
{
|
2007-08-20 22:15:58 -05:00
|
|
|
moo_dmsg ("try_send: file %s doesn't exist", filename);
|
2007-07-10 10:32:29 -05:00
|
|
|
goto out;
|
2005-06-22 18:20:32 +00:00
|
|
|
}
|
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
result = do_send (filename, data, data_len);
|
2006-12-19 02:12:37 -06:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
out:
|
|
|
|
g_free (filename);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
_moo_app_input_send_msg (const char *appname,
|
|
|
|
const char *name,
|
|
|
|
const char *data,
|
|
|
|
gssize len)
|
|
|
|
{
|
|
|
|
const char *entry;
|
|
|
|
GDir *pipe_dir = NULL;
|
|
|
|
char *pipe_dir_name;
|
|
|
|
gboolean success = FALSE;
|
|
|
|
|
|
|
|
g_return_val_if_fail (appname != NULL, FALSE);
|
|
|
|
g_return_val_if_fail (data != NULL, FALSE);
|
|
|
|
|
2007-08-20 22:15:58 -05:00
|
|
|
moo_dmsg ("_moo_app_input_send_msg: sending data to %s", name ? name : "NONE");
|
2007-07-10 10:32:29 -05:00
|
|
|
|
|
|
|
pipe_dir_name = get_pipe_dir (appname);
|
|
|
|
g_return_val_if_fail (pipe_dir_name != NULL, FALSE);
|
|
|
|
|
|
|
|
if (name)
|
2005-06-22 18:20:32 +00:00
|
|
|
{
|
2007-07-10 10:32:29 -05:00
|
|
|
success = try_send (pipe_dir_name, name, data, len);
|
|
|
|
goto out;
|
|
|
|
}
|
2005-06-22 18:20:32 +00:00
|
|
|
|
2007-07-31 08:02:07 -05:00
|
|
|
success = try_send (pipe_dir_name, MOO_APP_INPUT_NAME_DEFAULT, data, len);
|
|
|
|
if (success)
|
|
|
|
goto out;
|
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
pipe_dir = g_dir_open (pipe_dir_name, 0, NULL);
|
2005-06-22 18:20:32 +00:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
if (!pipe_dir)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
while ((entry = g_dir_read_name (pipe_dir)))
|
|
|
|
{
|
|
|
|
if (!strncmp (entry, INPUT_PREFIX, strlen (INPUT_PREFIX)))
|
2005-06-22 18:20:32 +00:00
|
|
|
{
|
2007-07-10 10:32:29 -05:00
|
|
|
name = entry + strlen (INPUT_PREFIX);
|
2006-03-16 04:39:35 -06:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
if (try_send (pipe_dir_name, name, data, len))
|
2005-06-22 18:20:32 +00:00
|
|
|
{
|
2007-07-10 10:32:29 -05:00
|
|
|
success = TRUE;
|
|
|
|
goto out;
|
2005-06-22 18:20:32 +00:00
|
|
|
}
|
|
|
|
}
|
2007-07-10 10:32:29 -05:00
|
|
|
}
|
2005-06-22 18:20:32 +00:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
out:
|
|
|
|
if (pipe_dir)
|
|
|
|
g_dir_close (pipe_dir);
|
|
|
|
g_free (pipe_dir_name);
|
|
|
|
return success;
|
|
|
|
}
|
2005-06-22 18:20:32 +00:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
static gboolean
|
|
|
|
do_write (int fd,
|
|
|
|
const char *data,
|
|
|
|
gsize data_len)
|
|
|
|
{
|
|
|
|
while (data_len > 0)
|
|
|
|
{
|
|
|
|
ssize_t n;
|
|
|
|
|
|
|
|
errno = 0;
|
|
|
|
n = write (fd, data, data_len);
|
|
|
|
|
|
|
|
if (n < 0)
|
2005-06-22 18:20:32 +00:00
|
|
|
{
|
2007-07-10 10:32:29 -05:00
|
|
|
if (errno != EAGAIN && errno != EINTR)
|
2005-06-22 18:20:32 +00:00
|
|
|
{
|
2007-07-10 10:32:29 -05:00
|
|
|
g_warning ("%s in write: %s", G_STRLOC, g_strerror (errno));
|
|
|
|
return FALSE;
|
2005-06-22 18:20:32 +00:00
|
|
|
}
|
|
|
|
}
|
2007-07-10 10:32:29 -05:00
|
|
|
else
|
|
|
|
{
|
|
|
|
data += n;
|
|
|
|
data_len -= n;
|
|
|
|
}
|
2005-06-22 18:20:32 +00:00
|
|
|
}
|
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
return TRUE;
|
2006-12-19 02:12:37 -06:00
|
|
|
}
|
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
#endif
|
2006-12-19 02:12:37 -06:00
|
|
|
|
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
#ifdef MOO_APP_INPUT_SOCKET
|
|
|
|
|
|
|
|
#ifndef UNIX_PATH_MAX
|
|
|
|
#define UNIX_PATH_MAX 108
|
|
|
|
#endif
|
2006-12-19 02:12:37 -06:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
typedef struct {
|
|
|
|
int fd;
|
|
|
|
GIOChannel *io;
|
|
|
|
guint io_watch;
|
|
|
|
GString *buffer; /* messages are zero-terminated */
|
|
|
|
InputChannel *ch;
|
|
|
|
} Connection;
|
|
|
|
|
|
|
|
struct InputChannel
|
|
|
|
{
|
|
|
|
char *name;
|
|
|
|
char *path;
|
|
|
|
char *pipe_dir;
|
|
|
|
gboolean owns_file;
|
|
|
|
int fd;
|
|
|
|
GIOChannel *io;
|
|
|
|
guint io_watch;
|
|
|
|
GSList *connections;
|
|
|
|
};
|
2006-12-19 02:12:37 -06:00
|
|
|
|
2007-07-13 03:42:31 -05:00
|
|
|
static char *
|
|
|
|
input_channel_get_path (InputChannel *ch)
|
|
|
|
{
|
|
|
|
g_return_val_if_fail (ch != NULL, NULL);
|
|
|
|
return g_strdup (ch->path);
|
|
|
|
}
|
|
|
|
|
2006-12-19 02:12:37 -06:00
|
|
|
static void
|
2007-07-10 10:32:29 -05:00
|
|
|
connection_free (Connection *conn)
|
2006-12-19 02:12:37 -06:00
|
|
|
{
|
2007-07-10 10:32:29 -05:00
|
|
|
if (conn->io_watch)
|
|
|
|
g_source_remove (conn->io_watch);
|
|
|
|
|
|
|
|
if (conn->io)
|
2006-12-19 02:12:37 -06:00
|
|
|
{
|
2007-07-10 10:32:29 -05:00
|
|
|
g_io_channel_shutdown (conn->io, FALSE, NULL);
|
|
|
|
g_io_channel_unref (conn->io);
|
|
|
|
}
|
2006-12-19 02:12:37 -06:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
if (conn->fd != -1)
|
|
|
|
close (conn->fd);
|
2006-12-19 02:12:37 -06:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
g_string_free (conn->buffer, TRUE);
|
|
|
|
_moo_free (Connection, conn);
|
2006-12-19 02:12:37 -06:00
|
|
|
}
|
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
static void
|
|
|
|
input_channel_shutdown (InputChannel *ch)
|
2006-12-19 02:12:37 -06:00
|
|
|
{
|
2007-07-10 10:32:29 -05:00
|
|
|
g_slist_foreach (ch->connections, (GFunc) connection_free, NULL);
|
|
|
|
g_slist_free (ch->connections);
|
|
|
|
ch->connections = NULL;
|
2006-12-19 02:12:37 -06:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
if (ch->io_watch)
|
|
|
|
{
|
|
|
|
g_source_remove (ch->io_watch);
|
|
|
|
ch->io_watch = 0;
|
|
|
|
}
|
2006-12-19 02:12:37 -06:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
if (ch->io)
|
|
|
|
{
|
|
|
|
g_io_channel_shutdown (ch->io, FALSE, NULL);
|
|
|
|
g_io_channel_unref (ch->io);
|
|
|
|
ch->io = NULL;
|
|
|
|
}
|
2006-12-19 02:12:37 -06:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
if (ch->fd != -1)
|
2006-12-19 02:12:37 -06:00
|
|
|
{
|
2007-07-10 10:32:29 -05:00
|
|
|
close (ch->fd);
|
|
|
|
ch->fd = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ch->path)
|
|
|
|
{
|
|
|
|
if (ch->owns_file)
|
|
|
|
unlink (ch->path);
|
|
|
|
g_free (ch->path);
|
|
|
|
ch->path = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
read_input (G_GNUC_UNUSED GIOChannel *source,
|
|
|
|
G_GNUC_UNUSED GIOCondition condition,
|
|
|
|
Connection *conn)
|
|
|
|
{
|
|
|
|
char c;
|
2007-07-31 08:02:07 -05:00
|
|
|
int n;
|
2007-07-15 03:04:05 -05:00
|
|
|
gboolean do_commit = FALSE;
|
2007-07-31 08:02:07 -05:00
|
|
|
|
2007-07-15 03:04:05 -05:00
|
|
|
errno = 0;
|
|
|
|
|
2007-07-31 08:02:07 -05:00
|
|
|
while ((n = read (conn->fd, &c, 1)) > 0)
|
2007-07-10 10:32:29 -05:00
|
|
|
{
|
|
|
|
if (c == 0)
|
2007-07-31 08:02:07 -05:00
|
|
|
{
|
2007-07-15 03:04:05 -05:00
|
|
|
do_commit = TRUE;
|
2007-07-31 08:02:07 -05:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
g_string_append_c (conn->buffer, c);
|
2007-07-10 10:32:29 -05:00
|
|
|
}
|
|
|
|
|
2007-07-31 08:02:07 -05:00
|
|
|
if (n <= 0)
|
2007-07-10 10:32:29 -05:00
|
|
|
{
|
2007-07-15 03:04:05 -05:00
|
|
|
if (n < 0)
|
2007-08-20 22:15:58 -05:00
|
|
|
moo_dmsg ("%s: %s", G_STRLOC, g_strerror (errno));
|
2007-07-15 03:04:05 -05:00
|
|
|
else
|
2007-08-20 22:15:58 -05:00
|
|
|
moo_dmsg ("%s: EOF", G_STRLOC);
|
2007-07-31 08:02:07 -05:00
|
|
|
goto remove;
|
2006-12-19 02:12:37 -06:00
|
|
|
}
|
2007-09-08 18:08:10 -05:00
|
|
|
else
|
|
|
|
{
|
|
|
|
moo_dmsg ("%s: got bytes: '%s'", G_STRLOC, conn->buffer->str);
|
|
|
|
}
|
2006-12-19 02:12:37 -06:00
|
|
|
|
2007-07-15 03:04:05 -05:00
|
|
|
if (do_commit)
|
|
|
|
commit (&conn->buffer);
|
|
|
|
|
2007-09-08 18:08:10 -05:00
|
|
|
if (condition & (G_IO_ERR | G_IO_HUP))
|
|
|
|
{
|
|
|
|
moo_dmsg ("%s: %s", G_STRLOC,
|
|
|
|
(condition & G_IO_ERR) ? "G_IO_ERR" : "G_IO_HUP");
|
|
|
|
goto remove;
|
|
|
|
}
|
|
|
|
|
2006-12-19 02:12:37 -06:00
|
|
|
return TRUE;
|
2007-07-31 08:02:07 -05:00
|
|
|
|
|
|
|
remove:
|
|
|
|
conn->ch->connections = g_slist_remove (conn->ch->connections, conn);
|
|
|
|
connection_free (conn);
|
|
|
|
return FALSE;
|
2006-12-19 02:12:37 -06:00
|
|
|
}
|
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
static gboolean
|
|
|
|
accept_connection (G_GNUC_UNUSED GIOChannel *source,
|
|
|
|
GIOCondition condition,
|
|
|
|
InputChannel *ch)
|
2006-12-19 02:12:37 -06:00
|
|
|
{
|
2007-07-10 10:32:29 -05:00
|
|
|
Connection *conn;
|
|
|
|
socklen_t dummy;
|
2006-12-19 02:12:37 -06:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
if (condition & G_IO_ERR)
|
|
|
|
{
|
|
|
|
input_channel_shutdown (ch);
|
|
|
|
return FALSE;
|
|
|
|
}
|
2006-12-19 02:12:37 -06:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
conn = _moo_new0 (Connection);
|
|
|
|
conn->ch = ch;
|
|
|
|
conn->buffer = g_string_new_len (NULL, MAX_BUFFER_SIZE);
|
2006-12-19 02:12:37 -06:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
conn->fd = accept (ch->fd, NULL, &dummy);
|
|
|
|
|
|
|
|
if (conn->fd == -1)
|
|
|
|
{
|
|
|
|
g_warning ("%s in accept: %s", G_STRLOC, g_strerror (errno));
|
|
|
|
_moo_free (Connection, conn);
|
2006-12-19 02:12:37 -06:00
|
|
|
return TRUE;
|
2007-07-10 10:32:29 -05:00
|
|
|
}
|
2006-12-19 02:12:37 -06:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
if (!input_channel_start_io (conn->fd, (GIOFunc) read_input, conn,
|
|
|
|
&conn->io, &conn->io_watch))
|
|
|
|
{
|
|
|
|
close (conn->fd);
|
|
|
|
_moo_free (Connection, conn);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
ch->connections = g_slist_prepend (ch->connections, conn);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2007-07-31 08:02:07 -05:00
|
|
|
static gboolean
|
|
|
|
try_connect (const char *filename,
|
|
|
|
int *fdp)
|
|
|
|
{
|
|
|
|
int fd;
|
|
|
|
struct sockaddr_un addr;
|
|
|
|
|
|
|
|
g_return_val_if_fail (filename != NULL, FALSE);
|
|
|
|
|
2007-10-26 04:46:57 -05:00
|
|
|
if (strlen (filename) + 1 > sizeof addr.sun_path)
|
2007-07-31 08:02:07 -05:00
|
|
|
{
|
|
|
|
g_critical ("%s: oops", G_STRLOC);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
addr.sun_family = AF_UNIX;
|
2007-10-26 04:46:57 -05:00
|
|
|
strcpy (addr.sun_path, filename);
|
2007-07-31 08:02:07 -05:00
|
|
|
fd = socket (PF_UNIX, SOCK_STREAM, 0);
|
|
|
|
|
|
|
|
if (fd == -1)
|
|
|
|
{
|
|
|
|
g_warning ("%s in socket for %s: %s", G_STRLOC, filename, g_strerror (errno));
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
errno = 0;
|
|
|
|
|
|
|
|
if (connect (fd, (struct sockaddr *) &addr, sizeof addr) == -1)
|
|
|
|
{
|
|
|
|
unlink (filename);
|
|
|
|
close (fd);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fdp)
|
|
|
|
*fdp = fd;
|
|
|
|
else
|
|
|
|
close (fd);
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
static gboolean
|
|
|
|
input_channel_start (InputChannel *ch,
|
|
|
|
gboolean may_fail)
|
|
|
|
{
|
|
|
|
struct sockaddr_un addr;
|
|
|
|
|
|
|
|
mkdir (ch->pipe_dir, S_IRWXU);
|
|
|
|
|
2007-07-31 08:02:07 -05:00
|
|
|
if (try_connect (ch->path, NULL))
|
|
|
|
{
|
|
|
|
if (!may_fail)
|
|
|
|
g_warning ("%s: '%s' is already in use",
|
|
|
|
G_STRLOC, ch->path);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2007-10-26 04:46:57 -05:00
|
|
|
if (strlen (ch->path) + 1 > sizeof addr.sun_path)
|
2007-07-10 10:32:29 -05:00
|
|
|
{
|
|
|
|
g_critical ("%s: oops", G_STRLOC);
|
2006-12-19 02:12:37 -06:00
|
|
|
return FALSE;
|
2007-07-10 10:32:29 -05:00
|
|
|
}
|
2006-12-19 02:12:37 -06:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
addr.sun_family = AF_UNIX;
|
2007-10-26 04:46:57 -05:00
|
|
|
strcpy (addr.sun_path, ch->path);
|
2006-12-19 02:12:37 -06:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
errno = 0;
|
|
|
|
|
|
|
|
if ((ch->fd = socket (PF_UNIX, SOCK_STREAM, 0)) == -1)
|
2006-12-19 02:12:37 -06:00
|
|
|
{
|
2007-07-10 10:32:29 -05:00
|
|
|
g_warning ("%s in socket for %s: %s", G_STRLOC, ch->path, g_strerror (errno));
|
|
|
|
return FALSE;
|
2006-12-19 02:12:37 -06:00
|
|
|
}
|
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
if (bind (ch->fd, (struct sockaddr*) &addr, sizeof addr) == -1)
|
2007-07-31 08:02:07 -05:00
|
|
|
{
|
|
|
|
g_warning ("%s in bind for %s: %s", G_STRLOC, ch->path, g_strerror (errno));
|
|
|
|
close (ch->fd);
|
|
|
|
ch->fd = -1;
|
|
|
|
return FALSE;
|
|
|
|
}
|
2007-07-10 10:32:29 -05:00
|
|
|
|
|
|
|
if (listen (ch->fd, 5) == -1)
|
2006-12-19 02:12:37 -06:00
|
|
|
{
|
2007-07-31 08:02:07 -05:00
|
|
|
g_warning ("%s in listen for %s: %s", G_STRLOC, ch->path, g_strerror (errno));
|
2007-07-10 10:32:29 -05:00
|
|
|
close (ch->fd);
|
|
|
|
ch->fd = -1;
|
|
|
|
return FALSE;
|
2006-12-19 02:12:37 -06:00
|
|
|
}
|
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
ch->owns_file = TRUE;
|
|
|
|
|
|
|
|
if (!input_channel_start_io (ch->fd, (GIOFunc) accept_connection, ch,
|
|
|
|
&ch->io, &ch->io_watch))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static InputChannel *
|
|
|
|
input_channel_new (const char *appname,
|
|
|
|
const char *name,
|
|
|
|
gboolean may_fail)
|
|
|
|
{
|
|
|
|
InputChannel *ch;
|
|
|
|
|
|
|
|
g_return_val_if_fail (appname != NULL, NULL);
|
|
|
|
g_return_val_if_fail (name != NULL, NULL);
|
|
|
|
|
|
|
|
ch = _moo_new0 (InputChannel);
|
|
|
|
|
|
|
|
ch->name = g_strdup (name);
|
|
|
|
ch->pipe_dir = get_pipe_dir (appname);
|
|
|
|
ch->path = get_pipe_path (ch->pipe_dir, name);
|
|
|
|
ch->fd = -1;
|
|
|
|
ch->io = NULL;
|
|
|
|
ch->io_watch = 0;
|
|
|
|
|
|
|
|
if (!input_channel_start (ch, may_fail))
|
2006-12-19 02:12:37 -06:00
|
|
|
{
|
2007-07-10 10:32:29 -05:00
|
|
|
input_channel_free (ch);
|
|
|
|
return NULL;
|
2006-12-19 02:12:37 -06:00
|
|
|
}
|
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
return ch;
|
|
|
|
}
|
2006-12-19 02:12:37 -06:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
static void
|
|
|
|
input_channel_free (InputChannel *ch)
|
|
|
|
{
|
|
|
|
input_channel_shutdown (ch);
|
|
|
|
g_free (ch->name);
|
|
|
|
g_free (ch->path);
|
2006-12-19 02:12:37 -06:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
if (ch->pipe_dir)
|
|
|
|
{
|
|
|
|
remove (ch->pipe_dir);
|
|
|
|
g_free (ch->pipe_dir);
|
|
|
|
}
|
|
|
|
|
|
|
|
_moo_free (InputChannel, ch);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
do_send (const char *filename,
|
|
|
|
const char *data,
|
|
|
|
gssize data_len)
|
|
|
|
{
|
|
|
|
int fd;
|
|
|
|
gboolean result = TRUE;
|
|
|
|
|
|
|
|
g_return_val_if_fail (filename != NULL, FALSE);
|
|
|
|
g_return_val_if_fail (data != NULL || data_len == 0, FALSE);
|
|
|
|
|
2007-07-31 08:02:07 -05:00
|
|
|
if (!try_connect (filename, &fd))
|
2007-07-10 10:32:29 -05:00
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (data_len < 0)
|
2007-07-31 08:02:07 -05:00
|
|
|
data_len = strlen (data) + 1;
|
2007-07-10 10:32:29 -05:00
|
|
|
|
|
|
|
if (data_len)
|
|
|
|
result = do_write (fd, data, data_len);
|
|
|
|
|
|
|
|
close (fd);
|
2006-12-19 02:12:37 -06:00
|
|
|
return result;
|
2005-06-22 18:20:32 +00:00
|
|
|
}
|
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
#endif /* MOO_APP_INPUT_SOCKET */
|
2005-06-22 18:20:32 +00:00
|
|
|
|
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
#ifdef MOO_APP_INPUT_PIPE
|
2005-06-22 18:20:32 +00:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
struct InputChannel
|
|
|
|
{
|
|
|
|
char *name;
|
|
|
|
char *path;
|
|
|
|
char *pipe_dir;
|
|
|
|
gboolean owns_file;
|
|
|
|
int fd;
|
|
|
|
GString *buffer;
|
|
|
|
GIOChannel *io;
|
|
|
|
guint io_watch;
|
|
|
|
GSList *connections;
|
|
|
|
};
|
|
|
|
|
2007-07-13 03:42:31 -05:00
|
|
|
static char *
|
|
|
|
input_channel_get_path (InputChannel *ch)
|
|
|
|
{
|
|
|
|
g_return_val_if_fail (ch != NULL, NULL);
|
|
|
|
return g_strdup (ch->path);
|
|
|
|
}
|
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
static void
|
|
|
|
input_channel_shutdown (InputChannel *ch)
|
|
|
|
{
|
|
|
|
if (ch->io_watch)
|
|
|
|
{
|
|
|
|
g_source_remove (ch->io_watch);
|
|
|
|
ch->io_watch = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ch->io)
|
|
|
|
{
|
|
|
|
g_io_channel_shutdown (ch->io, FALSE, NULL);
|
|
|
|
g_io_channel_unref (ch->io);
|
|
|
|
ch->io = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ch->fd != -1)
|
|
|
|
{
|
|
|
|
close (ch->fd);
|
|
|
|
ch->fd = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ch->path)
|
|
|
|
{
|
|
|
|
unlink (ch->path);
|
|
|
|
g_free (ch->path);
|
|
|
|
ch->path = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ch->buffer)
|
|
|
|
{
|
|
|
|
g_string_free (ch->buffer, TRUE);
|
|
|
|
ch->buffer = NULL;
|
|
|
|
}
|
|
|
|
}
|
2006-12-18 11:08:01 -06:00
|
|
|
|
2006-12-19 02:25:22 -06:00
|
|
|
static gboolean
|
|
|
|
read_input (GIOChannel *source,
|
|
|
|
GIOCondition condition,
|
2007-07-10 10:32:29 -05:00
|
|
|
InputChannel *ch)
|
2006-12-19 02:25:22 -06:00
|
|
|
{
|
|
|
|
gboolean error_occured = FALSE;
|
|
|
|
GError *err = NULL;
|
|
|
|
gboolean again = TRUE;
|
|
|
|
gboolean got_zero = FALSE;
|
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
g_return_val_if_fail (source == ch->io, FALSE);
|
2006-12-19 02:25:22 -06:00
|
|
|
|
|
|
|
/* XXX */
|
|
|
|
if (condition & (G_IO_ERR | G_IO_HUP))
|
|
|
|
error_occured = TRUE;
|
|
|
|
|
|
|
|
while (again && !error_occured && !err)
|
|
|
|
{
|
|
|
|
char c;
|
|
|
|
int bytes_read;
|
|
|
|
|
2007-01-11 20:28:22 -06:00
|
|
|
struct pollfd fd;
|
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
fd.fd = ch->fd;
|
2007-01-11 20:28:22 -06:00
|
|
|
fd.events = POLLIN | POLLPRI;
|
|
|
|
fd.revents = 0;
|
2006-12-19 02:25:22 -06:00
|
|
|
|
|
|
|
switch (poll (&fd, 1, 0))
|
|
|
|
{
|
|
|
|
case -1:
|
|
|
|
if (errno != EINTR && errno != EAGAIN)
|
|
|
|
error_occured = TRUE;
|
|
|
|
perror ("poll");
|
|
|
|
again = FALSE;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0:
|
|
|
|
if (0)
|
|
|
|
g_print ("%s: got nothing\n", G_STRLOC);
|
|
|
|
again = FALSE;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 1:
|
|
|
|
if (fd.revents & (POLLERR))
|
|
|
|
{
|
|
|
|
if (errno != EINTR && errno != EAGAIN)
|
|
|
|
error_occured = TRUE;
|
|
|
|
perror ("poll");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2007-07-10 10:32:29 -05:00
|
|
|
bytes_read = read (ch->fd, &c, 1);
|
2006-12-19 02:25:22 -06:00
|
|
|
|
|
|
|
if (bytes_read == 1)
|
|
|
|
{
|
2007-07-10 10:32:29 -05:00
|
|
|
g_string_append_c (ch->buffer, c);
|
2006-12-19 02:25:22 -06:00
|
|
|
|
|
|
|
if (!c)
|
|
|
|
{
|
|
|
|
got_zero = TRUE;
|
|
|
|
again = FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (bytes_read == -1)
|
|
|
|
{
|
|
|
|
perror ("read");
|
|
|
|
|
|
|
|
if (errno != EINTR && errno != EAGAIN)
|
|
|
|
error_occured = TRUE;
|
|
|
|
|
|
|
|
again = FALSE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
again = FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
g_assert_not_reached ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (error_occured || err)
|
|
|
|
{
|
|
|
|
g_critical ("%s: %s", G_STRLOC, err ? err->message : "error");
|
|
|
|
|
|
|
|
if (err)
|
|
|
|
g_error_free (err);
|
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
input_channel_shutdown (ch);
|
2006-12-19 02:25:22 -06:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (got_zero)
|
2007-07-10 10:32:29 -05:00
|
|
|
commit (&ch->buffer);
|
2006-12-19 02:25:22 -06:00
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
static gboolean
|
|
|
|
input_channel_start (InputChannel *ch,
|
|
|
|
G_GNUC_UNUSED gboolean may_fail)
|
2005-06-22 18:20:32 +00:00
|
|
|
{
|
2006-12-18 11:08:01 -06:00
|
|
|
mkdir (ch->pipe_dir, S_IRWXU);
|
2007-07-10 10:32:29 -05:00
|
|
|
unlink (ch->path);
|
2006-12-18 11:08:01 -06:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
if (mkfifo (ch->path, S_IRUSR | S_IWUSR) != 0)
|
2005-06-22 18:20:32 +00:00
|
|
|
{
|
|
|
|
int err = errno;
|
|
|
|
g_critical ("%s: error in mkfifo()", G_STRLOC);
|
|
|
|
g_critical ("%s: %s", G_STRLOC, g_strerror (err));
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2006-10-24 01:25:34 -05:00
|
|
|
/* XXX O_RDWR is not good (man 3p open), but it must be opened for
|
|
|
|
* writing by us, otherwise we get POLLHUP when a writer dies on the other end.
|
|
|
|
* So, open for writing separately. */
|
2007-07-10 10:32:29 -05:00
|
|
|
ch->fd = open (ch->path, O_RDWR | O_NONBLOCK);
|
|
|
|
if (ch->fd == -1)
|
2005-06-22 18:20:32 +00:00
|
|
|
{
|
|
|
|
int err = errno;
|
|
|
|
g_critical ("%s: error in open()", G_STRLOC);
|
|
|
|
g_critical ("%s: %s", G_STRLOC, g_strerror (err));
|
|
|
|
return FALSE;
|
|
|
|
}
|
2006-11-02 00:38:00 -06:00
|
|
|
|
2007-08-20 22:15:58 -05:00
|
|
|
moo_dmsg ("%s: opened input pipe %s with fd %d",
|
|
|
|
G_STRLOC, ch->path, ch->fd);
|
2005-06-22 18:20:32 +00:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
if (!input_channel_start_io (ch->fd, (GIOFunc) read_input, ch,
|
|
|
|
&ch->io, &ch->io_watch))
|
|
|
|
return FALSE;
|
2007-05-14 20:08:25 -05:00
|
|
|
|
2005-06-22 18:20:32 +00:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
static InputChannel *
|
|
|
|
input_channel_new (const char *appname,
|
|
|
|
const char *name,
|
|
|
|
gboolean may_fail)
|
|
|
|
{
|
|
|
|
InputChannel *ch;
|
|
|
|
|
|
|
|
g_return_val_if_fail (appname != NULL, NULL);
|
|
|
|
g_return_val_if_fail (name != NULL, NULL);
|
|
|
|
|
|
|
|
ch = _moo_new0 (InputChannel);
|
|
|
|
|
|
|
|
ch->name = g_strdup (name);
|
|
|
|
ch->pipe_dir = get_pipe_dir (appname);
|
|
|
|
ch->path = get_pipe_path (ch->pipe_dir, name);
|
|
|
|
ch->fd = -1;
|
|
|
|
ch->io = NULL;
|
|
|
|
ch->io_watch = 0;
|
|
|
|
ch->buffer = g_string_new_len (NULL, MAX_BUFFER_SIZE);
|
|
|
|
|
|
|
|
if (!input_channel_start (ch, may_fail))
|
|
|
|
{
|
|
|
|
input_channel_free (ch);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ch;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
input_channel_free (InputChannel *ch)
|
|
|
|
{
|
|
|
|
input_channel_shutdown (ch);
|
|
|
|
|
|
|
|
g_free (ch->name);
|
|
|
|
g_free (ch->path);
|
|
|
|
|
|
|
|
if (ch->pipe_dir)
|
|
|
|
{
|
|
|
|
remove (ch->pipe_dir);
|
|
|
|
g_free (ch->pipe_dir);
|
|
|
|
}
|
|
|
|
|
|
|
|
_moo_free (InputChannel, ch);
|
|
|
|
}
|
2005-06-22 18:20:32 +00:00
|
|
|
|
2006-08-18 02:24:18 -05:00
|
|
|
static gboolean
|
2007-07-10 10:32:29 -05:00
|
|
|
do_send (const char *filename,
|
|
|
|
const char *data,
|
|
|
|
gssize data_len)
|
2006-08-18 02:24:18 -05:00
|
|
|
{
|
|
|
|
gboolean result = FALSE;
|
2007-07-10 10:32:29 -05:00
|
|
|
int fd;
|
2006-08-18 02:24:18 -05:00
|
|
|
|
2007-08-20 22:15:58 -05:00
|
|
|
moo_dmsg ("do_send: sending data to `%s'", filename);
|
2006-08-18 02:24:18 -05:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
if (!g_file_test (filename, G_FILE_TEST_EXISTS))
|
|
|
|
{
|
2007-08-20 22:15:58 -05:00
|
|
|
moo_dmsg ("do_send: file `%s' doesn't exist", filename);
|
2007-07-10 10:32:29 -05:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
fd = open (filename, O_WRONLY | O_NONBLOCK);
|
2006-08-18 02:24:18 -05:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
if (fd == -1)
|
2006-08-22 23:19:08 -05:00
|
|
|
{
|
2007-08-20 22:15:58 -05:00
|
|
|
moo_dmsg ("do_send: could not open `%s': %s", filename, g_strerror (errno));
|
2007-07-10 10:32:29 -05:00
|
|
|
return FALSE;
|
2006-08-22 23:19:08 -05:00
|
|
|
}
|
2006-08-18 02:24:18 -05:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
result = do_write (fd, data, data_len);
|
|
|
|
close (fd);
|
2006-08-18 02:24:18 -05:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
if (result)
|
2007-08-20 22:15:58 -05:00
|
|
|
moo_dmsg ("do_send: successfully sent stuff to `%s'", filename);
|
2007-07-10 10:32:29 -05:00
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* MOO_APP_INPUT_PIPE */
|
|
|
|
|
|
|
|
|
|
|
|
/****************************************************************************/
|
|
|
|
/* WIN32
|
|
|
|
*/
|
|
|
|
#ifdef MOO_APP_INPUT_WIN32
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
char *pipe_name;
|
|
|
|
guint event_id;
|
|
|
|
} ListenerInfo;
|
|
|
|
|
|
|
|
struct InputChannel
|
|
|
|
{
|
|
|
|
char *appname;
|
|
|
|
char *name;
|
|
|
|
char *pipe_name;
|
|
|
|
GString *buffer;
|
|
|
|
guint event_id;
|
|
|
|
};
|
|
|
|
|
2007-07-13 03:42:31 -05:00
|
|
|
static char *
|
|
|
|
input_channel_get_path (InputChannel *ch)
|
|
|
|
{
|
|
|
|
g_return_val_if_fail (ch != NULL, NULL);
|
|
|
|
return g_strdup (ch->pipe_name);
|
|
|
|
}
|
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
static ListenerInfo *
|
|
|
|
listener_info_new (const char *pipe_name,
|
|
|
|
guint event_id)
|
|
|
|
{
|
|
|
|
ListenerInfo *info = g_new (ListenerInfo, 1);
|
|
|
|
info->pipe_name = g_strdup (pipe_name);
|
|
|
|
info->event_id = event_id;
|
|
|
|
return info;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
listener_info_free (ListenerInfo *info)
|
|
|
|
{
|
|
|
|
if (info)
|
2006-11-16 04:25:37 -06:00
|
|
|
{
|
2007-07-10 10:32:29 -05:00
|
|
|
g_free (info->pipe_name);
|
|
|
|
g_free (info);
|
2006-11-16 04:25:37 -06:00
|
|
|
}
|
2007-07-10 10:32:29 -05:00
|
|
|
}
|
2006-08-22 23:19:08 -05:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
|
|
|
|
static gpointer
|
|
|
|
listener_main (ListenerInfo *info)
|
|
|
|
{
|
|
|
|
HANDLE input;
|
|
|
|
|
|
|
|
_moo_message_async ("%s: hi there", G_STRLOC);
|
|
|
|
|
|
|
|
input = CreateNamedPipe (info->pipe_name,
|
|
|
|
PIPE_ACCESS_DUPLEX,
|
|
|
|
PIPE_TYPE_BYTE | PIPE_READMODE_BYTE | PIPE_WAIT,
|
|
|
|
PIPE_UNLIMITED_INSTANCES,
|
|
|
|
0, 0, 200, NULL);
|
|
|
|
|
|
|
|
if (input == INVALID_HANDLE_VALUE)
|
2006-08-18 02:24:18 -05:00
|
|
|
{
|
2007-07-10 10:32:29 -05:00
|
|
|
_moo_message_async ("%s: could not create input pipe", G_STRLOC);
|
|
|
|
listener_info_free (info);
|
|
|
|
return NULL;
|
2006-08-18 02:24:18 -05:00
|
|
|
}
|
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
_moo_message_async ("%s: opened pipe %s", G_STRLOC, info->pipe_name);
|
2006-08-18 02:24:18 -05:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
while (TRUE)
|
2006-11-16 04:25:37 -06:00
|
|
|
{
|
2007-07-10 10:32:29 -05:00
|
|
|
DWORD bytes_read;
|
|
|
|
char c;
|
|
|
|
|
|
|
|
DisconnectNamedPipe (input);
|
|
|
|
_moo_message_async ("%s: opening connection", G_STRLOC);
|
|
|
|
|
|
|
|
if (!ConnectNamedPipe (input, NULL))
|
|
|
|
{
|
|
|
|
DWORD err = GetLastError();
|
|
|
|
|
|
|
|
if (err != ERROR_PIPE_CONNECTED)
|
|
|
|
{
|
|
|
|
char *msg = g_win32_error_message (err);
|
|
|
|
_moo_message_async ("%s: error in ConnectNamedPipe()", G_STRLOC);
|
|
|
|
_moo_message_async ("%s: %s", G_STRLOC, msg);
|
|
|
|
CloseHandle (input);
|
|
|
|
g_free (msg);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
_moo_message_async ("%s: client connected", G_STRLOC);
|
|
|
|
|
|
|
|
while (ReadFile (input, &c, 1, &bytes_read, NULL))
|
|
|
|
{
|
|
|
|
if (bytes_read == 1)
|
|
|
|
{
|
|
|
|
_moo_event_queue_push (info->event_id, GINT_TO_POINTER ((int) c), NULL);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
_moo_message_async ("%s: client disconnected", G_STRLOC);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
_moo_message_async ("%s: goodbye", G_STRLOC);
|
|
|
|
|
|
|
|
CloseHandle (input);
|
|
|
|
listener_info_free (info);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static char *
|
|
|
|
get_pipe_name (const char *appname,
|
|
|
|
const char *name)
|
|
|
|
{
|
|
|
|
if (!name)
|
|
|
|
name = _moo_get_pid_string ();
|
|
|
|
return g_strdup_printf ("\\\\.\\pipe\\%s_in_%s",
|
|
|
|
appname, name);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
event_callback (GList *events,
|
|
|
|
InputChannel *ch)
|
|
|
|
{
|
|
|
|
while (events)
|
|
|
|
{
|
|
|
|
char c = GPOINTER_TO_INT (events->data);
|
|
|
|
|
|
|
|
if (c != 0)
|
|
|
|
g_string_append_c (ch->buffer, c);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gdk_threads_enter ();
|
|
|
|
commit (&ch->buffer);
|
|
|
|
gdk_threads_leave ();
|
|
|
|
}
|
|
|
|
|
|
|
|
events = events->next;
|
2006-11-16 04:25:37 -06:00
|
|
|
}
|
2007-07-10 10:32:29 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
input_channel_start (InputChannel *ch)
|
|
|
|
{
|
|
|
|
ListenerInfo *info;
|
2006-08-18 02:24:18 -05:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
g_free (ch->pipe_name);
|
|
|
|
ch->pipe_name = get_pipe_name (ch->appname, ch->name);
|
|
|
|
ch->event_id = _moo_event_queue_connect ((MooEventQueueCallback) event_callback,
|
|
|
|
ch, NULL);
|
|
|
|
|
|
|
|
info = listener_info_new (ch->pipe_name, ch->event_id);
|
2006-08-18 02:24:18 -05:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
if (!g_thread_create ((GThreadFunc) listener_main, info, FALSE, NULL))
|
2006-11-16 04:25:37 -06:00
|
|
|
{
|
2007-07-10 10:32:29 -05:00
|
|
|
g_critical ("could not start listener thread");
|
|
|
|
listener_info_free (info);
|
|
|
|
g_free (ch->pipe_name);
|
|
|
|
ch->pipe_name = NULL;
|
|
|
|
_moo_event_queue_disconnect (ch->event_id);
|
|
|
|
ch->event_id = 0;
|
|
|
|
return FALSE;
|
2006-11-16 04:25:37 -06:00
|
|
|
}
|
2006-08-18 02:24:18 -05:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static InputChannel *
|
|
|
|
input_channel_new (const char *appname,
|
|
|
|
const char *name,
|
|
|
|
G_GNUC_UNUSED gboolean may_fail)
|
|
|
|
{
|
|
|
|
InputChannel *ch;
|
2006-08-18 02:24:18 -05:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
ch = _moo_new0 (InputChannel);
|
|
|
|
ch->appname = g_strdup (appname);
|
|
|
|
ch->name = g_strdup (name);
|
|
|
|
ch->buffer = g_string_new_len (NULL, MAX_BUFFER_SIZE);
|
|
|
|
|
|
|
|
if (!input_channel_start (ch))
|
2006-11-16 04:25:37 -06:00
|
|
|
{
|
2007-07-10 10:32:29 -05:00
|
|
|
input_channel_free (ch);
|
|
|
|
return NULL;
|
2006-11-16 04:25:37 -06:00
|
|
|
}
|
2006-08-18 02:24:18 -05:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
return ch;
|
|
|
|
}
|
2006-08-18 02:24:18 -05:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
static void
|
|
|
|
input_channel_free (InputChannel *ch)
|
|
|
|
{
|
|
|
|
if (ch->event_id)
|
|
|
|
_moo_event_queue_disconnect (ch->event_id);
|
|
|
|
if (ch->buffer)
|
|
|
|
g_string_free (ch->buffer, TRUE);
|
|
|
|
g_free (ch->pipe_name);
|
|
|
|
g_free (ch->appname);
|
|
|
|
g_free (ch->name);
|
|
|
|
_moo_free (InputChannel, ch);
|
2006-08-18 02:24:18 -05:00
|
|
|
}
|
|
|
|
|
2005-11-02 02:32:07 +00:00
|
|
|
gboolean
|
2007-07-10 10:32:29 -05:00
|
|
|
_moo_app_input_send_msg (const char *appname,
|
|
|
|
const char *name,
|
2006-12-18 11:08:01 -06:00
|
|
|
const char *data,
|
2007-07-10 10:32:29 -05:00
|
|
|
gssize len)
|
2005-11-02 02:32:07 +00:00
|
|
|
{
|
2007-07-10 10:32:29 -05:00
|
|
|
char *err_msg = NULL;
|
|
|
|
char *pipe_name;
|
|
|
|
HANDLE pipe_handle;
|
|
|
|
gboolean result = FALSE;
|
|
|
|
DWORD bytes_written;
|
2005-11-02 02:32:07 +00:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
g_return_val_if_fail (appname != NULL, FALSE);
|
2005-11-02 02:32:07 +00:00
|
|
|
g_return_val_if_fail (data != NULL, FALSE);
|
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
if (len < 0)
|
|
|
|
len = strlen (data);
|
2006-11-16 04:25:37 -06:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
if (!len)
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
if (!name)
|
|
|
|
name = "main";
|
|
|
|
|
|
|
|
pipe_name = get_pipe_name (appname, name);
|
|
|
|
pipe_handle = CreateFile (pipe_name, GENERIC_WRITE, 0, NULL, OPEN_EXISTING, 0, NULL);
|
2005-11-02 02:32:07 +00:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
if (!pipe_handle)
|
2005-11-02 02:32:07 +00:00
|
|
|
{
|
2007-07-10 10:32:29 -05:00
|
|
|
err_msg = g_win32_error_message (GetLastError ());
|
|
|
|
g_warning ("could not open pipe '%s': %s", pipe_name, err_msg);
|
2006-08-18 02:24:18 -05:00
|
|
|
goto out;
|
|
|
|
}
|
2005-11-02 02:32:07 +00:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
if (!WriteFile (pipe_handle, data, len, &bytes_written, NULL))
|
|
|
|
{
|
|
|
|
err_msg = g_win32_error_message (GetLastError ());
|
|
|
|
g_warning ("could not write data to '%s': %s", pipe_name, err_msg);
|
2006-08-18 02:24:18 -05:00
|
|
|
goto out;
|
2007-07-10 10:32:29 -05:00
|
|
|
}
|
2005-11-02 02:32:07 +00:00
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
if (bytes_written < (DWORD) len)
|
2006-08-18 02:24:18 -05:00
|
|
|
{
|
2007-07-10 10:32:29 -05:00
|
|
|
g_warning ("written less data than requested to '%s'", pipe_name);
|
|
|
|
goto out;
|
2005-11-02 02:32:07 +00:00
|
|
|
}
|
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
result = TRUE;
|
|
|
|
|
2005-11-02 02:32:07 +00:00
|
|
|
out:
|
2007-07-10 10:32:29 -05:00
|
|
|
if (pipe_handle != INVALID_HANDLE_VALUE)
|
|
|
|
CloseHandle (pipe_handle);
|
|
|
|
|
|
|
|
g_free (pipe_name);
|
|
|
|
g_free (err_msg);
|
|
|
|
return result;
|
2005-11-02 02:32:07 +00:00
|
|
|
}
|
|
|
|
|
2007-07-10 10:32:29 -05:00
|
|
|
#endif /* MOO_APP_INPUT_WIN32 */
|