blob: 028569a8904a33ae9b912b2a0fece00fe4abf4c8 [file] [log] [blame]
#include "slp_server.hpp"
#include "sock_channel.hpp"
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <memory>
/** General udp server which waits for the POLLIN event
on the port and calls the call back once it gets the event.
usage would be create the server with the port and the call back
and call the run method.
*/
int slp::udp::Server::run()
{
struct sockaddr_in6 serverAddr{};
sd_event* event = nullptr;
slp::deleted_unique_ptr<sd_event> eventPtr(event, [](sd_event* event) {
if (!event)
{
event = sd_event_unref(event);
}
});
int fd = -1, r;
sigset_t ss;
r = sd_event_default(&event);
if (r < 0)
{
goto finish;
}
eventPtr.reset(event);
event = nullptr;
if (sigemptyset(&ss) < 0 || sigaddset(&ss, SIGTERM) < 0 ||
sigaddset(&ss, SIGINT) < 0)
{
r = -errno;
goto finish;
}
/* Block SIGTERM first, so that the event loop can handle it */
if (sigprocmask(SIG_BLOCK, &ss, NULL) < 0)
{
r = -errno;
goto finish;
}
/* Let's make use of the default handler and "floating"
reference features of sd_event_add_signal() */
r = sd_event_add_signal(eventPtr.get(), NULL, SIGTERM, NULL, NULL);
if (r < 0)
{
goto finish;
}
r = sd_event_add_signal(eventPtr.get(), NULL, SIGINT, NULL, NULL);
if (r < 0)
{
goto finish;
}
fd = socket(AF_INET6, SOCK_DGRAM | SOCK_CLOEXEC | SOCK_NONBLOCK, 0);
if (fd < 0)
{
r = -errno;
goto finish;
}
serverAddr.sin6_family = AF_INET6;
serverAddr.sin6_port = htons(this->port);
if (bind(fd, (struct sockaddr*)&serverAddr, sizeof(serverAddr)) < 0)
{
r = -errno;
goto finish;
}
r = sd_event_add_io(eventPtr.get(), nullptr, fd, EPOLLIN, this->callme,
nullptr);
if (r < 0)
{
goto finish;
}
r = sd_event_loop(eventPtr.get());
finish:
if (fd >= 0)
{
(void)close(fd);
}
if (r < 0)
{
fprintf(stderr, "Failure: %s\n", strerror(-r));
}
return r < 0 ? EXIT_FAILURE : EXIT_SUCCESS;
}