mirror of
https://salsa.debian.org/ha-team/libqb
synced 2025-08-26 05:57:31 +00:00

Fixes the following warnings under clang: loop_poll_kqueue.c:58:2: warning: incompatible integer to pointer conversion assigning to 'void *' from 'intptr_t' (aka 'long') [-Wint-conversion] EV_SET(&ke, fd, filters, EV_ADD | EV_ENABLE, 0, 0, (intptr_t)pe); ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ /usr/include/sys/event.h:54:16: note: expanded from macro 'EV_SET' (kevp)->udata = (f); \ ^ ~~~
213 lines
4.9 KiB
C
213 lines
4.9 KiB
C
/*
|
|
* Copyright (C) 2012 Red Hat, Inc.
|
|
*
|
|
* Author: Angus Salkeld <asalkeld@redhat.com>
|
|
*
|
|
* This file is part of libqb.
|
|
*
|
|
* libqb 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.
|
|
*
|
|
* libqb 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 libqb. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "os_base.h"
|
|
#include "loop_poll_int.h"
|
|
|
|
#ifdef HAVE_SYS_EVENT_H
|
|
#include <sys/event.h>
|
|
#endif /* HAVE_SYS_EVENT_H */
|
|
|
|
#define MAX_EVENTS 12
|
|
|
|
static int32_t
|
|
_poll_to_filter_(int32_t event)
|
|
{
|
|
int32_t out = 0;
|
|
if (event & POLLIN)
|
|
out |= EVFILT_READ;
|
|
if (event & POLLOUT)
|
|
out |= EVFILT_WRITE;
|
|
return out;
|
|
}
|
|
|
|
static void
|
|
_fini(struct qb_poll_source *s)
|
|
{
|
|
if (s->epollfd != -1) {
|
|
close(s->epollfd);
|
|
s->epollfd = -1;
|
|
}
|
|
}
|
|
|
|
static int32_t
|
|
_add(struct qb_poll_source *s, struct qb_poll_entry *pe, int32_t fd, int32_t events)
|
|
{
|
|
int32_t res = 0;
|
|
struct kevent ke;
|
|
short filters = _poll_to_filter_(events);
|
|
|
|
EV_SET(&ke, fd, filters, EV_ADD | EV_ENABLE, 0, 0, pe);
|
|
|
|
res = kevent(s->epollfd, &ke, 1, NULL, 0, NULL);
|
|
if (res == -1) {
|
|
res = -errno;
|
|
qb_util_perror(LOG_ERR, "kevent(add)");
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
static int32_t
|
|
_mod(struct qb_poll_source *s, struct qb_poll_entry *pe, int32_t fd, int32_t events)
|
|
{
|
|
int32_t res = 0;
|
|
struct kevent ke[2];
|
|
short new_filters = _poll_to_filter_(events);
|
|
short old_filters = _poll_to_filter_(pe->ufd.events);
|
|
|
|
EV_SET(&ke[0], fd, old_filters, EV_DELETE, 0, 0, pe);
|
|
EV_SET(&ke[1], fd, new_filters, EV_ADD | EV_ENABLE, 0, 0, pe);
|
|
|
|
res = kevent(s->epollfd, ke, 2, NULL, 0, NULL);
|
|
if (res == -1) {
|
|
res = -errno;
|
|
qb_util_perror(LOG_ERR, "kevent(mod)");
|
|
}
|
|
return res;
|
|
}
|
|
|
|
static int32_t
|
|
_del(struct qb_poll_source *s, struct qb_poll_entry *pe, int32_t fd, int32_t events)
|
|
{
|
|
int32_t res = 0;
|
|
struct kevent ke;
|
|
short filters = _poll_to_filter_(events);
|
|
|
|
EV_SET(&ke, fd, filters, EV_DELETE, 0, 0, pe);
|
|
|
|
res = kevent(s->epollfd, &ke, 1, NULL, 0, NULL);
|
|
if (res == -1 && errno == ENOENT) {
|
|
/*
|
|
* Not a problem already cleaned up.
|
|
*/
|
|
res = 0;
|
|
} else if (res == -1) {
|
|
res = -errno;
|
|
qb_util_perror(LOG_ERR, "kevent(del)");
|
|
}
|
|
return res;
|
|
}
|
|
|
|
static int32_t
|
|
_poll_and_add_to_jobs_(struct qb_loop_source *src, int32_t ms_timeout)
|
|
{
|
|
int32_t i;
|
|
int32_t event_count;
|
|
int32_t new_jobs = 0;
|
|
int32_t revents = 0;
|
|
struct qb_poll_entry *pe = NULL;
|
|
struct qb_poll_source *s = (struct qb_poll_source *)src;
|
|
struct kevent events[MAX_EVENTS];
|
|
struct timespec timeout = { 0, 0 };
|
|
struct timespec *timeout_pt = &timeout;
|
|
|
|
if (ms_timeout > 0) {
|
|
qb_timespec_add_ms(&timeout, ms_timeout);
|
|
} else if (ms_timeout < 0) {
|
|
timeout_pt = NULL;
|
|
}
|
|
qb_poll_fds_usage_check_(s);
|
|
|
|
retry_poll:
|
|
|
|
event_count = kevent(s->epollfd, NULL, 0, events, MAX_EVENTS, timeout_pt);
|
|
if (errno == EINTR && event_count == -1) {
|
|
goto retry_poll;
|
|
} else if (event_count == -1) {
|
|
qb_util_perror(LOG_ERR, "kevent(poll)");
|
|
return -errno;
|
|
}
|
|
|
|
for (i = 0; i < event_count; i++) {
|
|
revents = 0;
|
|
pe = (struct qb_poll_entry *)events[i].udata;
|
|
#if 0
|
|
if (events[i].flags) {
|
|
qb_util_log(LOG_TRACE,
|
|
"got flags %d on fd %d.", events[i].flags,
|
|
pe->ufd.fd);
|
|
}
|
|
#endif
|
|
if (events[i].flags & EV_ERROR) {
|
|
qb_util_log(LOG_WARNING,
|
|
"got EV_ERROR on fd %d.", pe->ufd.fd);
|
|
revents |= POLLERR;
|
|
}
|
|
if (events[i].flags & EV_EOF) {
|
|
qb_util_log(LOG_INFO,
|
|
"got EV_EOF on fd %d.", pe->ufd.fd);
|
|
revents |= POLLHUP;
|
|
}
|
|
if (events[i].filter == EVFILT_READ) {
|
|
revents |= POLLIN;
|
|
}
|
|
if (events[i].filter == EVFILT_WRITE) {
|
|
revents |= POLLOUT;
|
|
}
|
|
if (pe->ufd.fd == -1 || pe->state == QB_POLL_ENTRY_DELETED) {
|
|
qb_util_log(LOG_WARNING,
|
|
"can't post new event to a deleted entry.");
|
|
/*
|
|
* empty/deleted
|
|
*/
|
|
EV_SET(&events[i], events[i].ident, events[i].filter,
|
|
EV_DELETE, 0, 0, pe);
|
|
(void)kevent(s->epollfd, &events[i], 1, NULL, 0, NULL);
|
|
continue;
|
|
}
|
|
if (pe->ufd.fd != events[i].ident) {
|
|
qb_util_log(LOG_WARNING,
|
|
"can't find poll entry for new event.");
|
|
continue;
|
|
}
|
|
if (revents == pe->ufd.revents ||
|
|
pe->state == QB_POLL_ENTRY_JOBLIST) {
|
|
/*
|
|
* entry already in the job queue.
|
|
*/
|
|
continue;
|
|
}
|
|
pe->ufd.revents = revents;
|
|
|
|
new_jobs += pe->add_to_jobs(src->l, pe);
|
|
}
|
|
|
|
return new_jobs;
|
|
}
|
|
|
|
int32_t
|
|
qb_kqueue_init(struct qb_poll_source *s)
|
|
{
|
|
s->epollfd = kqueue();
|
|
|
|
if (s->epollfd < 0) {
|
|
qb_util_perror(LOG_ERR, "kqueue()");
|
|
return -errno;
|
|
}
|
|
s->driver.fini = _fini;
|
|
s->driver.add = _add;
|
|
s->driver.mod = _mod;
|
|
s->driver.del = _del;
|
|
s->s.poll = _poll_and_add_to_jobs_;
|
|
return 0;
|
|
}
|