Blame examples/ipcserver.c

Packit Service f88c7f
/*
Packit Service f88c7f
 * Copyright (c) 2011 Red Hat, Inc.
Packit Service f88c7f
 *
Packit Service f88c7f
 * All rights reserved.
Packit Service f88c7f
 *
Packit Service f88c7f
 * Author: Angus Salkeld <asalkeld@redhat.com>
Packit Service f88c7f
 *
Packit Service f88c7f
 * libqb is free software: you can redistribute it and/or modify
Packit Service f88c7f
 * it under the terms of the GNU Lesser General Public License as published by
Packit Service f88c7f
 * the Free Software Foundation, either version 2.1 of the License, or
Packit Service f88c7f
 * (at your option) any later version.
Packit Service f88c7f
 *
Packit Service f88c7f
 * libqb is distributed in the hope that it will be useful,
Packit Service f88c7f
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service f88c7f
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit Service f88c7f
 * GNU Lesser General Public License for more details.
Packit Service f88c7f
 *
Packit Service f88c7f
 * You should have received a copy of the GNU Lesser General Public License
Packit Service f88c7f
 * along with libqb.  If not, see <http://www.gnu.org/licenses/>.
Packit Service f88c7f
 */
Packit Service f88c7f
#include "os_base.h"
Packit Service f88c7f
#include <signal.h>
Packit Service f88c7f
Packit Service f88c7f
#include <qb/qbarray.h>
Packit Service f88c7f
#include <qb/qbdefs.h>
Packit Service f88c7f
#include <qb/qbutil.h>
Packit Service f88c7f
#include <qb/qblog.h>
Packit Service f88c7f
#include <qb/qbloop.h>
Packit Service f88c7f
#include <qb/qbipcs.h>
Packit Service f88c7f
Packit Service f88c7f
#ifdef HAVE_GLIB
Packit Service f88c7f
#include <glib.h>
Packit Service f88c7f
static GMainLoop *glib_loop;
Packit Service f88c7f
static qb_array_t *gio_map;
Packit Service f88c7f
#endif /* HAVE_GLIB */
Packit Service f88c7f
Packit Service f88c7f
#define ONE_MEG 1048576
Packit Service f88c7f
Packit Service f88c7f
static int32_t use_glib = QB_FALSE;
Packit Service f88c7f
static int32_t use_events = QB_FALSE;
Packit Service f88c7f
static qb_loop_t *bms_loop;
Packit Service f88c7f
static qb_ipcs_service_t *s1;
Packit Service f88c7f
Packit Service f88c7f
static int32_t
Packit Service f88c7f
s1_connection_accept_fn(qb_ipcs_connection_t * c, uid_t uid, gid_t gid)
Packit Service f88c7f
{
Packit Service f88c7f
#if 0
Packit Service f88c7f
	if (uid == 0 && gid == 0) {
Packit Service f88c7f
		qb_log(LOG_INFO, "Authenticated connection");
Packit Service f88c7f
		return 1;
Packit Service f88c7f
	}
Packit Service f88c7f
	qb_log(LOG_NOTICE, "BAD user!");
Packit Service f88c7f
	return 0;
Packit Service f88c7f
#else
Packit Service f88c7f
	return 0;
Packit Service f88c7f
#endif
Packit Service f88c7f
}
Packit Service f88c7f
Packit Service f88c7f
static void
Packit Service f88c7f
s1_connection_created_fn(qb_ipcs_connection_t * c)
Packit Service f88c7f
{
Packit Service f88c7f
	struct qb_ipcs_stats srv_stats;
Packit Service f88c7f
Packit Service f88c7f
	qb_ipcs_stats_get(s1, &srv_stats, QB_FALSE);
Packit Service f88c7f
	qb_log(LOG_INFO, "Connection created (active:%d, closed:%d)",
Packit Service f88c7f
	       srv_stats.active_connections, srv_stats.closed_connections);
Packit Service f88c7f
}
Packit Service f88c7f
Packit Service f88c7f
static void
Packit Service f88c7f
s1_connection_destroyed_fn(qb_ipcs_connection_t * c)
Packit Service f88c7f
{
Packit Service f88c7f
	qb_log(LOG_INFO, "Connection about to be freed");
Packit Service f88c7f
}
Packit Service f88c7f
Packit Service f88c7f
static int32_t
Packit Service f88c7f
s1_connection_closed_fn(qb_ipcs_connection_t * c)
Packit Service f88c7f
{
Packit Service f88c7f
	struct qb_ipcs_connection_stats stats;
Packit Service f88c7f
	struct qb_ipcs_stats srv_stats;
Packit Service f88c7f
Packit Service f88c7f
	qb_ipcs_stats_get(s1, &srv_stats, QB_FALSE);
Packit Service f88c7f
	qb_ipcs_connection_stats_get(c, &stats, QB_FALSE);
Packit Service f88c7f
	qb_log(LOG_INFO,
Packit Service f88c7f
	       "Connection to pid:%d destroyed (active:%d, closed:%d)",
Packit Service f88c7f
	       stats.client_pid, srv_stats.active_connections,
Packit Service f88c7f
	       srv_stats.closed_connections);
Packit Service f88c7f
Packit Service f88c7f
	qb_log(LOG_DEBUG, " Requests     %"PRIu64"", stats.requests);
Packit Service f88c7f
	qb_log(LOG_DEBUG, " Responses    %"PRIu64"", stats.responses);
Packit Service f88c7f
	qb_log(LOG_DEBUG, " Events       %"PRIu64"", stats.events);
Packit Service f88c7f
	qb_log(LOG_DEBUG, " Send retries %"PRIu64"", stats.send_retries);
Packit Service f88c7f
	qb_log(LOG_DEBUG, " Recv retries %"PRIu64"", stats.recv_retries);
Packit Service f88c7f
	qb_log(LOG_DEBUG, " FC state     %d", stats.flow_control_state);
Packit Service f88c7f
	qb_log(LOG_DEBUG, " FC count     %"PRIu64"", stats.flow_control_count);
Packit Service f88c7f
	return 0;
Packit Service f88c7f
}
Packit Service f88c7f
Packit Service f88c7f
struct my_req {
Packit Service f88c7f
	struct qb_ipc_request_header hdr;
Packit Service f88c7f
	char message[256];
Packit Service f88c7f
};
Packit Service f88c7f
Packit Service f88c7f
static int32_t
Packit Service f88c7f
s1_msg_process_fn(qb_ipcs_connection_t * c, void *data, size_t size)
Packit Service f88c7f
{
Packit Service f88c7f
	struct qb_ipc_request_header *hdr;
Packit Service f88c7f
	struct my_req *req_pt;
Packit Service f88c7f
	struct qb_ipc_response_header response;
Packit Service f88c7f
	ssize_t res;
Packit Service f88c7f
	struct iovec iov[2];
Packit Service f88c7f
	char resp[100];
Packit Service f88c7f
	int32_t sl;
Packit Service f88c7f
	int32_t send_ten_events = QB_FALSE;
Packit Service f88c7f
Packit Service f88c7f
	hdr = (struct qb_ipc_request_header *)data;
Packit Service f88c7f
	if (hdr->id == (QB_IPC_MSG_USER_START + 1)) {
Packit Service f88c7f
		return 0;
Packit Service f88c7f
	}
Packit Service f88c7f
Packit Service f88c7f
	req_pt = (struct my_req *)data;
Packit Service f88c7f
	qb_log(LOG_DEBUG, "msg received (id:%d, size:%d, data:%s)",
Packit Service f88c7f
	       req_pt->hdr.id, req_pt->hdr.size, req_pt->message);
Packit Service f88c7f
Packit Service f88c7f
	if (strcmp(req_pt->message, "kill") == 0) {
Packit Service f88c7f
		exit(0);
Packit Service f88c7f
	}
Packit Service f88c7f
	response.size = sizeof(struct qb_ipc_response_header);
Packit Service f88c7f
	response.id = 13;
Packit Service f88c7f
	response.error = 0;
Packit Service f88c7f
Packit Service f88c7f
	sl = snprintf(resp, 100, "ACK %zu bytes", size) + 1;
Packit Service f88c7f
	iov[0].iov_len = sizeof(response);
Packit Service f88c7f
	iov[0].iov_base = &response;
Packit Service f88c7f
	iov[1].iov_len = sl;
Packit Service f88c7f
	iov[1].iov_base = resp;
Packit Service f88c7f
	response.size += sl;
Packit Service f88c7f
Packit Service f88c7f
	send_ten_events = (strcmp(req_pt->message, "events") == 0);
Packit Service f88c7f
Packit Service f88c7f
	if (use_events && !send_ten_events) {
Packit Service f88c7f
		res = qb_ipcs_event_sendv(c, iov, 2);
Packit Service f88c7f
	} else {
Packit Service f88c7f
		res = qb_ipcs_response_sendv(c, iov, 2);
Packit Service f88c7f
	}
Packit Service f88c7f
	if (res < 0) {
Packit Service f88c7f
		errno = - res;
Packit Service f88c7f
		qb_perror(LOG_ERR, "qb_ipcs_response_send");
Packit Service f88c7f
	}
Packit Service f88c7f
	if (send_ten_events) {
Packit Service f88c7f
		int32_t i;
Packit Service f88c7f
		qb_log(LOG_INFO, "request to send 10 events");
Packit Service f88c7f
		for (i = 0; i < 10; i++) {
Packit Service f88c7f
			res = qb_ipcs_event_sendv(c, iov, 2);
Packit Service f88c7f
			qb_log(LOG_INFO, "sent event %d res:%d", i, res);
Packit Service f88c7f
		}
Packit Service f88c7f
	}
Packit Service f88c7f
	return 0;
Packit Service f88c7f
}
Packit Service f88c7f
Packit Service f88c7f
static void
Packit Service f88c7f
sigusr1_handler(int32_t num)
Packit Service f88c7f
{
Packit Service f88c7f
	qb_log(LOG_DEBUG, "(%d)", num);
Packit Service f88c7f
	qb_ipcs_destroy(s1);
Packit Service f88c7f
	exit(0);
Packit Service f88c7f
}
Packit Service f88c7f
Packit Service f88c7f
static void
Packit Service f88c7f
show_usage(const char *name)
Packit Service f88c7f
{
Packit Service f88c7f
	printf("usage: \n");
Packit Service f88c7f
	printf("%s <options>\n", name);
Packit Service f88c7f
	printf("\n");
Packit Service f88c7f
	printf("  options:\n");
Packit Service f88c7f
	printf("\n");
Packit Service f88c7f
	printf("  -h             show this help text\n");
Packit Service f88c7f
	printf("  -m             use shared memory\n");
Packit Service f88c7f
	printf("  -u             use unix sockets\n");
Packit Service f88c7f
	printf("  -g             use glib mainloop\n");
Packit Service f88c7f
	printf("  -e             use events\n");
Packit Service f88c7f
	printf("\n");
Packit Service f88c7f
}
Packit Service f88c7f
Packit Service f88c7f
#ifdef HAVE_GLIB
Packit Service f88c7f
struct gio_to_qb_poll {
Packit Service f88c7f
	int32_t is_used;
Packit Service f88c7f
	int32_t events;
Packit Service f88c7f
	int32_t source;
Packit Service f88c7f
	int32_t fd;
Packit Service f88c7f
	void *data;
Packit Service f88c7f
	qb_ipcs_dispatch_fn_t fn;
Packit Service f88c7f
	enum qb_loop_priority p;
Packit Service f88c7f
};
Packit Service f88c7f
Packit Service f88c7f
static gboolean
Packit Service f88c7f
gio_read_socket(GIOChannel * gio, GIOCondition condition, gpointer data)
Packit Service f88c7f
{
Packit Service f88c7f
	struct gio_to_qb_poll *adaptor = (struct gio_to_qb_poll *)data;
Packit Service f88c7f
	gint fd = g_io_channel_unix_get_fd(gio);
Packit Service f88c7f
Packit Service f88c7f
	return (adaptor->fn(fd, condition, adaptor->data) == 0);
Packit Service f88c7f
}
Packit Service f88c7f
Packit Service f88c7f
static void
Packit Service f88c7f
gio_poll_destroy(gpointer data)
Packit Service f88c7f
{
Packit Service f88c7f
	struct gio_to_qb_poll *adaptor = (struct gio_to_qb_poll *)data;
Packit Service f88c7f
Packit Service f88c7f
	adaptor->is_used--;
Packit Service f88c7f
	if (adaptor->is_used == 0) {
Packit Service f88c7f
		qb_log(LOG_DEBUG, "fd %d adaptor destroyed\n", adaptor->fd);
Packit Service f88c7f
		adaptor->fd = 0;
Packit Service f88c7f
		adaptor->source = 0;
Packit Service f88c7f
	}
Packit Service f88c7f
}
Packit Service f88c7f
Packit Service f88c7f
static int32_t
Packit Service f88c7f
my_g_dispatch_update(enum qb_loop_priority p, int32_t fd, int32_t evts,
Packit Service f88c7f
		  void *data, qb_ipcs_dispatch_fn_t fn, gboolean is_new)
Packit Service f88c7f
{
Packit Service f88c7f
	struct gio_to_qb_poll *adaptor;
Packit Service f88c7f
	GIOChannel *channel;
Packit Service f88c7f
	int32_t res = 0;
Packit Service f88c7f
Packit Service f88c7f
	res = qb_array_index(gio_map, fd, (void **)&adaptor);
Packit Service f88c7f
	if (res < 0) {
Packit Service f88c7f
		return res;
Packit Service f88c7f
	}
Packit Service f88c7f
	if (adaptor->is_used && adaptor->source) {
Packit Service f88c7f
		if (is_new) {
Packit Service f88c7f
			return -EEXIST;
Packit Service f88c7f
		}
Packit Service f88c7f
		g_source_remove(adaptor->source);
Packit Service f88c7f
		adaptor->source = 0;
Packit Service f88c7f
	}
Packit Service f88c7f
Packit Service f88c7f
	channel = g_io_channel_unix_new(fd);
Packit Service f88c7f
	if (!channel) {
Packit Service f88c7f
		return -ENOMEM;
Packit Service f88c7f
	}
Packit Service f88c7f
Packit Service f88c7f
	adaptor->fn = fn;
Packit Service f88c7f
	adaptor->events = evts;
Packit Service f88c7f
	adaptor->data = data;
Packit Service f88c7f
	adaptor->p = p;
Packit Service f88c7f
	adaptor->is_used++;
Packit Service f88c7f
	adaptor->fd = fd;
Packit Service f88c7f
Packit Service f88c7f
	adaptor->source = g_io_add_watch_full(channel, G_PRIORITY_DEFAULT, evts, gio_read_socket, adaptor, gio_poll_destroy);
Packit Service f88c7f
Packit Service f88c7f
	/* we are handing the channel off to be managed by mainloop now.
Packit Service f88c7f
	 * remove our reference. */
Packit Service f88c7f
	g_io_channel_unref(channel);
Packit Service f88c7f
Packit Service f88c7f
	return 0;
Packit Service f88c7f
}
Packit Service f88c7f
Packit Service f88c7f
static int32_t
Packit Service f88c7f
my_g_dispatch_add(enum qb_loop_priority p, int32_t fd, int32_t evts,
Packit Service f88c7f
		  void *data, qb_ipcs_dispatch_fn_t fn)
Packit Service f88c7f
{
Packit Service f88c7f
	return my_g_dispatch_update(p, fd, evts, data, fn, TRUE);
Packit Service f88c7f
}
Packit Service f88c7f
Packit Service f88c7f
static int32_t
Packit Service f88c7f
my_g_dispatch_mod(enum qb_loop_priority p, int32_t fd, int32_t evts,
Packit Service f88c7f
		  void *data, qb_ipcs_dispatch_fn_t fn)
Packit Service f88c7f
{
Packit Service f88c7f
	return my_g_dispatch_update(p, fd, evts, data, fn, FALSE);
Packit Service f88c7f
}
Packit Service f88c7f
Packit Service f88c7f
static int32_t
Packit Service f88c7f
my_g_dispatch_del(int32_t fd)
Packit Service f88c7f
{
Packit Service f88c7f
	struct gio_to_qb_poll *adaptor;
Packit Service f88c7f
	if (qb_array_index(gio_map, fd, (void **)&adaptor) == 0) {
Packit Service f88c7f
		g_source_remove(adaptor->source);
Packit Service f88c7f
		adaptor->source = 0;
Packit Service f88c7f
	}
Packit Service f88c7f
	return 0;
Packit Service f88c7f
}
Packit Service f88c7f
#endif /* HAVE_GLIB */
Packit Service f88c7f
Packit Service f88c7f
static int32_t
Packit Service f88c7f
my_job_add(enum qb_loop_priority p, void *data, qb_loop_job_dispatch_fn fn)
Packit Service f88c7f
{
Packit Service f88c7f
	return qb_loop_job_add(bms_loop, p, data, fn);
Packit Service f88c7f
}
Packit Service f88c7f
Packit Service f88c7f
static int32_t
Packit Service f88c7f
my_dispatch_add(enum qb_loop_priority p, int32_t fd, int32_t evts,
Packit Service f88c7f
		void *data, qb_ipcs_dispatch_fn_t fn)
Packit Service f88c7f
{
Packit Service f88c7f
	return qb_loop_poll_add(bms_loop, p, fd, evts, data, fn);
Packit Service f88c7f
}
Packit Service f88c7f
Packit Service f88c7f
static int32_t
Packit Service f88c7f
my_dispatch_mod(enum qb_loop_priority p, int32_t fd, int32_t evts,
Packit Service f88c7f
		void *data, qb_ipcs_dispatch_fn_t fn)
Packit Service f88c7f
{
Packit Service f88c7f
	return qb_loop_poll_mod(bms_loop, p, fd, evts, data, fn);
Packit Service f88c7f
}
Packit Service f88c7f
Packit Service f88c7f
static int32_t
Packit Service f88c7f
my_dispatch_del(int32_t fd)
Packit Service f88c7f
{
Packit Service f88c7f
	return qb_loop_poll_del(bms_loop, fd);
Packit Service f88c7f
}
Packit Service f88c7f
Packit Service f88c7f
int32_t
Packit Service f88c7f
main(int32_t argc, char *argv[])
Packit Service f88c7f
{
Packit Service f88c7f
	const char *options = "mpseugh";
Packit Service f88c7f
	int32_t opt;
Packit Service f88c7f
	int32_t rc;
Packit Service f88c7f
	enum qb_ipc_type ipc_type = QB_IPC_NATIVE;
Packit Service f88c7f
	struct qb_ipcs_service_handlers sh = {
Packit Service f88c7f
		.connection_accept = s1_connection_accept_fn,
Packit Service f88c7f
		.connection_created = s1_connection_created_fn,
Packit Service f88c7f
		.msg_process = s1_msg_process_fn,
Packit Service f88c7f
		.connection_destroyed = s1_connection_destroyed_fn,
Packit Service f88c7f
		.connection_closed = s1_connection_closed_fn,
Packit Service f88c7f
	};
Packit Service f88c7f
	struct qb_ipcs_poll_handlers ph = {
Packit Service f88c7f
		.job_add = my_job_add,
Packit Service f88c7f
		.dispatch_add = my_dispatch_add,
Packit Service f88c7f
		.dispatch_mod = my_dispatch_mod,
Packit Service f88c7f
		.dispatch_del = my_dispatch_del,
Packit Service f88c7f
	};
Packit Service f88c7f
#ifdef HAVE_GLIB
Packit Service f88c7f
	struct qb_ipcs_poll_handlers glib_ph = {
Packit Service f88c7f
		.job_add = NULL, /* FIXME */
Packit Service f88c7f
		.dispatch_add = my_g_dispatch_add,
Packit Service f88c7f
		.dispatch_mod = my_g_dispatch_mod,
Packit Service f88c7f
		.dispatch_del = my_g_dispatch_del,
Packit Service f88c7f
	};
Packit Service f88c7f
#endif /* HAVE_GLIB */
Packit Service f88c7f
Packit Service f88c7f
	while ((opt = getopt(argc, argv, options)) != -1) {
Packit Service f88c7f
		switch (opt) {
Packit Service f88c7f
		case 'm':
Packit Service f88c7f
			ipc_type = QB_IPC_SHM;
Packit Service f88c7f
			break;
Packit Service f88c7f
		case 'u':
Packit Service f88c7f
			ipc_type = QB_IPC_SOCKET;
Packit Service f88c7f
			break;
Packit Service f88c7f
		case 'g':
Packit Service f88c7f
			use_glib = QB_TRUE;
Packit Service f88c7f
			break;
Packit Service f88c7f
		case 'e':
Packit Service f88c7f
			use_events = QB_TRUE;
Packit Service f88c7f
			break;
Packit Service f88c7f
		case 'h':
Packit Service f88c7f
		default:
Packit Service f88c7f
			show_usage(argv[0]);
Packit Service f88c7f
			exit(0);
Packit Service f88c7f
			break;
Packit Service f88c7f
		}
Packit Service f88c7f
	}
Packit Service f88c7f
	signal(SIGINT, sigusr1_handler);
Packit Service f88c7f
Packit Service f88c7f
	qb_log_init("ipcserver", LOG_USER, LOG_TRACE);
Packit Service f88c7f
	qb_log_filter_ctl(QB_LOG_STDERR, QB_LOG_FILTER_ADD,
Packit Service f88c7f
			  QB_LOG_FILTER_FILE, "*", LOG_TRACE);
Packit Service f88c7f
	qb_log_format_set(QB_LOG_STDERR, "%f:%l [%p] %b");
Packit Service f88c7f
	qb_log_ctl(QB_LOG_STDERR, QB_LOG_CONF_ENABLED, QB_TRUE);
Packit Service f88c7f
Packit Service f88c7f
	s1 = qb_ipcs_create("ipcserver", 0, ipc_type, &sh);
Packit Service f88c7f
	if (s1 == 0) {
Packit Service f88c7f
		qb_perror(LOG_ERR, "qb_ipcs_create");
Packit Service f88c7f
		exit(1);
Packit Service f88c7f
	}
Packit Service f88c7f
	/* This forces the clients to use a minimum buffer size */
Packit Service f88c7f
	qb_ipcs_enforce_buffer_size(s1, ONE_MEG);
Packit Service f88c7f
Packit Service f88c7f
	if (!use_glib) {
Packit Service f88c7f
		bms_loop = qb_loop_create();
Packit Service f88c7f
		qb_ipcs_poll_handlers_set(s1, &ph);
Packit Service f88c7f
		rc = qb_ipcs_run(s1);
Packit Service f88c7f
		if (rc != 0) {
Packit Service f88c7f
			errno = -rc;
Packit Service f88c7f
			qb_perror(LOG_ERR, "qb_ipcs_run");
Packit Service f88c7f
			exit(1);
Packit Service f88c7f
		}
Packit Service f88c7f
		qb_loop_run(bms_loop);
Packit Service f88c7f
	} else {
Packit Service f88c7f
#ifdef HAVE_GLIB
Packit Service f88c7f
		glib_loop = g_main_loop_new(NULL, FALSE);
Packit Service f88c7f
		gio_map = qb_array_create_2(16, sizeof(struct gio_to_qb_poll), 1);
Packit Service f88c7f
		qb_ipcs_poll_handlers_set(s1, &glib_ph);
Packit Service f88c7f
		rc = qb_ipcs_run(s1);
Packit Service f88c7f
		if (rc != 0) {
Packit Service f88c7f
			errno = -rc;
Packit Service f88c7f
			qb_perror(LOG_ERR, "qb_ipcs_run");
Packit Service f88c7f
			exit(1);
Packit Service f88c7f
		}
Packit Service f88c7f
		g_main_loop_run(glib_loop);
Packit Service f88c7f
#else
Packit Service f88c7f
		qb_log(LOG_ERR,
Packit Service f88c7f
		       "You don't seem to have glib-devel installed.\n");
Packit Service f88c7f
#endif
Packit Service f88c7f
	}
Packit Service f88c7f
	qb_log_fini();
Packit Service f88c7f
	return EXIT_SUCCESS;
Packit Service f88c7f
}