/*
This file is part of Ingen.
Copyright 2007-2012 David Robillard
Ingen is free software: you can redistribute it and/or modify it under the
terms of the GNU Affero General Public License as published by the Free
Software Foundation, either version 3 of the License, or any later version.
Ingen 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 Affero General Public License for details.
You should have received a copy of the GNU Affero General Public License
along with Ingen. If not, see .
*/
#include
#include
#include
#include "ingen/AtomReader.hpp"
#include "ingen/Configuration.hpp"
#include "ingen/Interface.hpp"
#include "ingen/Log.hpp"
#include "ingen/World.hpp"
#include "sord/sordmm.hpp"
#include "sratom/sratom.h"
#include "../server/Engine.hpp"
#include "../server/EventWriter.hpp"
#include "SocketListener.hpp"
#include "SocketServer.hpp"
namespace Ingen {
namespace Socket {
SocketListener::SocketListener(Ingen::World& world)
: Raul::Thread()
, _world(world)
, _unix_sock(Socket::UNIX)
, _net_sock(Socket::TCP)
{
// Create UNIX socket
_unix_path = world.conf().option("socket").get_string();
const Raul::URI unix_uri("unix://" + _unix_path);
if (!_unix_sock.bind(unix_uri) || !_unix_sock.listen()) {
_world.log().error("Failed to create UNIX socket\n");
_unix_sock.close();
}
_world.log().info(Raul::fmt("Listening on socket %1%\n") % unix_uri);
// Create TCP socket
int port = world.conf().option("engine-port").get_int32();
std::ostringstream ss;
ss << "tcp://localhost:";
ss << port;
if (!_net_sock.bind(Raul::URI(ss.str())) || !_net_sock.listen()) {
_world.log().error("Failed to create TCP socket\n");
_net_sock.close();
}
_world.log().info(Raul::fmt("Listening on TCP port %1%\n") % port);
start();
}
SocketListener::~SocketListener()
{
_exit_flag = true;
_unix_sock.shutdown();
_net_sock.shutdown();
join();
_unix_sock.close();
_net_sock.close();
unlink(_unix_path.c_str());
}
void
SocketListener::_run()
{
Server::Engine* engine = (Server::Engine*)_world.engine().get();
struct pollfd pfds[2];
int nfds = 0;
if (_unix_sock.fd() != -1) {
pfds[nfds].fd = _unix_sock.fd();
pfds[nfds].events = POLLIN;
pfds[nfds].revents = 0;
++nfds;
}
if (_net_sock.fd() != -1) {
pfds[nfds].fd = _net_sock.fd();
pfds[nfds].events = POLLIN;
pfds[nfds].revents = 0;
++nfds;
}
while (true) {
// Wait for input to arrive at a socket
int ret = poll(pfds, nfds, -1);
if (_exit_flag) {
break;
} else if (ret == -1) {
_world.log().error(Raul::fmt("Poll error: %1%\n") % strerror(errno));
break;
} else if (ret == 0) {
_world.log().error("Poll returned with no data\n");
continue;
}
if (pfds[0].revents & POLLIN) {
SharedPtr conn = _unix_sock.accept();
if (conn) {
new SocketServer(_world, *engine, conn);
}
}
if (pfds[1].revents & POLLIN) {
SharedPtr conn = _net_sock.accept();
if (conn) {
new SocketServer(_world, *engine, conn);
}
}
}
}
} // namespace Ingen
} // namespace Socket