Blame src/shared/mainloop.c

Packit 34410b
/*
Packit 34410b
 *
Packit 34410b
 *  BlueZ - Bluetooth protocol stack for Linux
Packit 34410b
 *
Packit 34410b
 *  Copyright (C) 2011-2014  Intel Corporation
Packit 34410b
 *  Copyright (C) 2002-2010  Marcel Holtmann <marcel@holtmann.org>
Packit 34410b
 *
Packit 34410b
 *
Packit 34410b
 *  This library is free software; you can redistribute it and/or
Packit 34410b
 *  modify it under the terms of the GNU Lesser General Public
Packit 34410b
 *  License as published by the Free Software Foundation; either
Packit 34410b
 *  version 2.1 of the License, or (at your option) any later version.
Packit 34410b
 *
Packit 34410b
 *  This library is distributed in the hope that it will be useful,
Packit 34410b
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 34410b
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 34410b
 *  Lesser General Public License for more details.
Packit 34410b
 *
Packit 34410b
 *  You should have received a copy of the GNU Lesser General Public
Packit 34410b
 *  License along with this library; if not, write to the Free Software
Packit 34410b
 *  Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
Packit 34410b
 *
Packit 34410b
 */
Packit 34410b
Packit 34410b
#ifdef HAVE_CONFIG_H
Packit 34410b
#include <config.h>
Packit 34410b
#endif
Packit 34410b
Packit 34410b
#define _GNU_SOURCE
Packit 34410b
#include <stdio.h>
Packit 34410b
#include <errno.h>
Packit 34410b
#include <unistd.h>
Packit 34410b
#include <stdlib.h>
Packit 34410b
#include <stddef.h>
Packit 34410b
#include <string.h>
Packit 34410b
#include <signal.h>
Packit 34410b
#include <sys/signalfd.h>
Packit 34410b
#include <sys/timerfd.h>
Packit 34410b
#include <sys/epoll.h>
Packit 34410b
#include <sys/socket.h>
Packit 34410b
#include <sys/un.h>
Packit 34410b
Packit 34410b
#include "mainloop.h"
Packit 34410b
#include "mainloop-notify.h"
Packit 34410b
Packit 34410b
#define MAX_EPOLL_EVENTS 10
Packit 34410b
Packit 34410b
static int epoll_fd;
Packit 34410b
static int epoll_terminate;
Packit 34410b
static int exit_status = EXIT_SUCCESS;
Packit 34410b
Packit 34410b
struct mainloop_data {
Packit 34410b
	int fd;
Packit 34410b
	uint32_t events;
Packit 34410b
	mainloop_event_func callback;
Packit 34410b
	mainloop_destroy_func destroy;
Packit 34410b
	void *user_data;
Packit 34410b
};
Packit 34410b
Packit 34410b
#define MAX_MAINLOOP_ENTRIES 128
Packit 34410b
Packit 34410b
static struct mainloop_data *mainloop_list[MAX_MAINLOOP_ENTRIES];
Packit 34410b
Packit 34410b
struct timeout_data {
Packit 34410b
	int fd;
Packit 34410b
	mainloop_timeout_func callback;
Packit 34410b
	mainloop_destroy_func destroy;
Packit 34410b
	void *user_data;
Packit 34410b
};
Packit 34410b
Packit 34410b
void mainloop_init(void)
Packit 34410b
{
Packit 34410b
	unsigned int i;
Packit 34410b
Packit 34410b
	epoll_fd = epoll_create1(EPOLL_CLOEXEC);
Packit 34410b
Packit 34410b
	for (i = 0; i < MAX_MAINLOOP_ENTRIES; i++)
Packit 34410b
		mainloop_list[i] = NULL;
Packit 34410b
Packit 34410b
	epoll_terminate = 0;
Packit 34410b
Packit 34410b
	mainloop_notify_init();
Packit 34410b
}
Packit 34410b
Packit 34410b
void mainloop_quit(void)
Packit 34410b
{
Packit 34410b
	epoll_terminate = 1;
Packit 34410b
Packit 34410b
	mainloop_sd_notify("STOPPING=1");
Packit 34410b
}
Packit 34410b
Packit 34410b
void mainloop_exit_success(void)
Packit 34410b
{
Packit 34410b
	exit_status = EXIT_SUCCESS;
Packit 34410b
	epoll_terminate = 1;
Packit 34410b
}
Packit 34410b
Packit 34410b
void mainloop_exit_failure(void)
Packit 34410b
{
Packit 34410b
	exit_status = EXIT_FAILURE;
Packit 34410b
	epoll_terminate = 1;
Packit 34410b
}
Packit 34410b
Packit 34410b
int mainloop_run(void)
Packit 34410b
{
Packit 34410b
	unsigned int i;
Packit 34410b
Packit 34410b
	while (!epoll_terminate) {
Packit 34410b
		struct epoll_event events[MAX_EPOLL_EVENTS];
Packit 34410b
		int n, nfds;
Packit 34410b
Packit 34410b
		nfds = epoll_wait(epoll_fd, events, MAX_EPOLL_EVENTS, -1);
Packit 34410b
		if (nfds < 0)
Packit 34410b
			continue;
Packit 34410b
Packit 34410b
		for (n = 0; n < nfds; n++) {
Packit 34410b
			struct mainloop_data *data = events[n].data.ptr;
Packit 34410b
Packit 34410b
			data->callback(data->fd, events[n].events,
Packit 34410b
							data->user_data);
Packit 34410b
		}
Packit 34410b
	}
Packit 34410b
Packit 34410b
	for (i = 0; i < MAX_MAINLOOP_ENTRIES; i++) {
Packit 34410b
		struct mainloop_data *data = mainloop_list[i];
Packit 34410b
Packit 34410b
		mainloop_list[i] = NULL;
Packit 34410b
Packit 34410b
		if (data) {
Packit 34410b
			epoll_ctl(epoll_fd, EPOLL_CTL_DEL, data->fd, NULL);
Packit 34410b
Packit 34410b
			if (data->destroy)
Packit 34410b
				data->destroy(data->user_data);
Packit 34410b
Packit 34410b
			free(data);
Packit 34410b
		}
Packit 34410b
	}
Packit 34410b
Packit 34410b
	close(epoll_fd);
Packit 34410b
	epoll_fd = 0;
Packit 34410b
Packit 34410b
	mainloop_notify_exit();
Packit 34410b
Packit 34410b
	return exit_status;
Packit 34410b
}
Packit 34410b
Packit 34410b
int mainloop_add_fd(int fd, uint32_t events, mainloop_event_func callback,
Packit 34410b
				void *user_data, mainloop_destroy_func destroy)
Packit 34410b
{
Packit 34410b
	struct mainloop_data *data;
Packit 34410b
	struct epoll_event ev;
Packit 34410b
	int err;
Packit 34410b
Packit 34410b
	if (fd < 0 || fd > MAX_MAINLOOP_ENTRIES - 1 || !callback)
Packit 34410b
		return -EINVAL;
Packit 34410b
Packit 34410b
	data = malloc(sizeof(*data));
Packit 34410b
	if (!data)
Packit 34410b
		return -ENOMEM;
Packit 34410b
Packit 34410b
	memset(data, 0, sizeof(*data));
Packit 34410b
	data->fd = fd;
Packit 34410b
	data->events = events;
Packit 34410b
	data->callback = callback;
Packit 34410b
	data->destroy = destroy;
Packit 34410b
	data->user_data = user_data;
Packit 34410b
Packit 34410b
	memset(&ev, 0, sizeof(ev));
Packit 34410b
	ev.events = events;
Packit 34410b
	ev.data.ptr = data;
Packit 34410b
Packit 34410b
	err = epoll_ctl(epoll_fd, EPOLL_CTL_ADD, data->fd, &ev;;
Packit 34410b
	if (err < 0) {
Packit 34410b
		free(data);
Packit 34410b
		return err;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	mainloop_list[fd] = data;
Packit 34410b
Packit 34410b
	return 0;
Packit 34410b
}
Packit 34410b
Packit 34410b
int mainloop_modify_fd(int fd, uint32_t events)
Packit 34410b
{
Packit 34410b
	struct mainloop_data *data;
Packit 34410b
	struct epoll_event ev;
Packit 34410b
	int err;
Packit 34410b
Packit 34410b
	if (fd < 0 || fd > MAX_MAINLOOP_ENTRIES - 1)
Packit 34410b
		return -EINVAL;
Packit 34410b
Packit 34410b
	data = mainloop_list[fd];
Packit 34410b
	if (!data)
Packit 34410b
		return -ENXIO;
Packit 34410b
Packit 34410b
	memset(&ev, 0, sizeof(ev));
Packit 34410b
	ev.events = events;
Packit 34410b
	ev.data.ptr = data;
Packit 34410b
Packit 34410b
	err = epoll_ctl(epoll_fd, EPOLL_CTL_MOD, data->fd, &ev;;
Packit 34410b
	if (err < 0)
Packit 34410b
		return err;
Packit 34410b
Packit 34410b
	data->events = events;
Packit 34410b
Packit 34410b
	return 0;
Packit 34410b
}
Packit 34410b
Packit 34410b
int mainloop_remove_fd(int fd)
Packit 34410b
{
Packit 34410b
	struct mainloop_data *data;
Packit 34410b
	int err;
Packit 34410b
Packit 34410b
	if (fd < 0 || fd > MAX_MAINLOOP_ENTRIES - 1)
Packit 34410b
		return -EINVAL;
Packit 34410b
Packit 34410b
	data = mainloop_list[fd];
Packit 34410b
	if (!data)
Packit 34410b
		return -ENXIO;
Packit 34410b
Packit 34410b
	mainloop_list[fd] = NULL;
Packit 34410b
Packit 34410b
	err = epoll_ctl(epoll_fd, EPOLL_CTL_DEL, data->fd, NULL);
Packit 34410b
Packit 34410b
	if (data->destroy)
Packit 34410b
		data->destroy(data->user_data);
Packit 34410b
Packit 34410b
	free(data);
Packit 34410b
Packit 34410b
	return err;
Packit 34410b
}
Packit 34410b
Packit 34410b
static void timeout_destroy(void *user_data)
Packit 34410b
{
Packit 34410b
	struct timeout_data *data = user_data;
Packit 34410b
Packit 34410b
	close(data->fd);
Packit 34410b
	data->fd = -1;
Packit 34410b
Packit 34410b
	if (data->destroy)
Packit 34410b
		data->destroy(data->user_data);
Packit 34410b
Packit 34410b
	free(data);
Packit 34410b
}
Packit 34410b
Packit 34410b
static void timeout_callback(int fd, uint32_t events, void *user_data)
Packit 34410b
{
Packit 34410b
	struct timeout_data *data = user_data;
Packit 34410b
	uint64_t expired;
Packit 34410b
	ssize_t result;
Packit 34410b
Packit 34410b
	if (events & (EPOLLERR | EPOLLHUP))
Packit 34410b
		return;
Packit 34410b
Packit 34410b
	result = read(data->fd, &expired, sizeof(expired));
Packit 34410b
	if (result != sizeof(expired))
Packit 34410b
		return;
Packit 34410b
Packit 34410b
	if (data->callback)
Packit 34410b
		data->callback(data->fd, data->user_data);
Packit 34410b
}
Packit 34410b
Packit 34410b
static inline int timeout_set(int fd, unsigned int msec)
Packit 34410b
{
Packit 34410b
	struct itimerspec itimer;
Packit 34410b
	unsigned int sec = msec / 1000;
Packit 34410b
Packit 34410b
	memset(&itimer, 0, sizeof(itimer));
Packit 34410b
	itimer.it_interval.tv_sec = 0;
Packit 34410b
	itimer.it_interval.tv_nsec = 0;
Packit 34410b
	itimer.it_value.tv_sec = sec;
Packit 34410b
	itimer.it_value.tv_nsec = (msec - (sec * 1000)) * 1000 * 1000;
Packit 34410b
Packit 34410b
	return timerfd_settime(fd, 0, &itimer, NULL);
Packit 34410b
}
Packit 34410b
Packit 34410b
int mainloop_add_timeout(unsigned int msec, mainloop_timeout_func callback,
Packit 34410b
				void *user_data, mainloop_destroy_func destroy)
Packit 34410b
{
Packit 34410b
	struct timeout_data *data;
Packit 34410b
Packit 34410b
	if (!callback)
Packit 34410b
		return -EINVAL;
Packit 34410b
Packit 34410b
	data = malloc(sizeof(*data));
Packit 34410b
	if (!data)
Packit 34410b
		return -ENOMEM;
Packit 34410b
Packit 34410b
	memset(data, 0, sizeof(*data));
Packit 34410b
	data->callback = callback;
Packit 34410b
	data->destroy = destroy;
Packit 34410b
	data->user_data = user_data;
Packit 34410b
Packit 34410b
	data->fd = timerfd_create(CLOCK_MONOTONIC, TFD_NONBLOCK | TFD_CLOEXEC);
Packit 34410b
	if (data->fd < 0) {
Packit 34410b
		free(data);
Packit 34410b
		return -EIO;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	if (msec > 0) {
Packit 34410b
		if (timeout_set(data->fd, msec) < 0) {
Packit 34410b
			close(data->fd);
Packit 34410b
			free(data);
Packit 34410b
			return -EIO;
Packit 34410b
		}
Packit 34410b
	}
Packit 34410b
Packit 34410b
	if (mainloop_add_fd(data->fd, EPOLLIN | EPOLLONESHOT,
Packit 34410b
				timeout_callback, data, timeout_destroy) < 0) {
Packit 34410b
		close(data->fd);
Packit 34410b
		free(data);
Packit 34410b
		return -EIO;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	return data->fd;
Packit 34410b
}
Packit 34410b
Packit 34410b
int mainloop_modify_timeout(int id, unsigned int msec)
Packit 34410b
{
Packit 34410b
	if (msec > 0) {
Packit 34410b
		if (timeout_set(id, msec) < 0)
Packit 34410b
			return -EIO;
Packit 34410b
	}
Packit 34410b
Packit 34410b
	if (mainloop_modify_fd(id, EPOLLIN | EPOLLONESHOT) < 0)
Packit 34410b
		return -EIO;
Packit 34410b
Packit 34410b
	return 0;
Packit 34410b
}
Packit 34410b
Packit 34410b
int mainloop_remove_timeout(int id)
Packit 34410b
{
Packit 34410b
	return mainloop_remove_fd(id);
Packit 34410b
}