2004-12-30 22:43:55 +00:00
|
|
|
#include "tools/socket.h"
|
2012-03-03 23:04:11 +01:00
|
|
|
#include "tools/select.h"
|
2004-12-30 22:43:55 +00:00
|
|
|
|
2009-02-13 10:39:20 +00:00
|
|
|
#include <vdr/tools.h>
|
2004-12-30 22:43:55 +00:00
|
|
|
#include <string.h>
|
|
|
|
#include <arpa/inet.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
|
2007-05-09 09:12:42 +00:00
|
|
|
// default class: best effort
|
|
|
|
#define DSCP_BE 0
|
|
|
|
// gold class (video): assured forwarding 4 with lowest drop precedence
|
|
|
|
#define DSCP_AF41 34 << 2
|
|
|
|
// premium class (voip): expedited forwarding
|
|
|
|
#define DSCP_EF 46 << 2
|
|
|
|
// actual DSCP value used
|
|
|
|
#define STREAMDEV_DSCP DSCP_AF41
|
|
|
|
|
2009-02-13 10:39:20 +00:00
|
|
|
cTBSocket::cTBSocket(int Type, int Protocol) {
|
2004-12-30 22:43:55 +00:00
|
|
|
memset(&m_LocalAddr, 0, sizeof(m_LocalAddr));
|
|
|
|
memset(&m_RemoteAddr, 0, sizeof(m_RemoteAddr));
|
|
|
|
m_Type = Type;
|
2009-02-13 10:39:20 +00:00
|
|
|
m_Protocol = Protocol;
|
2004-12-30 22:43:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
cTBSocket::~cTBSocket() {
|
|
|
|
if (IsOpen()) Close();
|
|
|
|
}
|
|
|
|
|
2012-03-03 23:04:11 +01:00
|
|
|
bool cTBSocket::Connect(const std::string &Host, unsigned int Port, unsigned int TimeoutMs) {
|
2004-12-30 22:43:55 +00:00
|
|
|
socklen_t len;
|
|
|
|
int socket;
|
|
|
|
|
|
|
|
if (IsOpen()) Close();
|
|
|
|
|
2009-02-13 10:39:20 +00:00
|
|
|
if ((socket = ::socket(PF_INET, m_Type, m_Protocol)) == -1)
|
2004-12-30 22:43:55 +00:00
|
|
|
return false;
|
|
|
|
|
|
|
|
m_LocalAddr.sin_family = AF_INET;
|
|
|
|
m_LocalAddr.sin_port = 0;
|
|
|
|
m_LocalAddr.sin_addr.s_addr = INADDR_ANY;
|
|
|
|
if (::bind(socket, (struct sockaddr*)&m_LocalAddr, sizeof(m_LocalAddr))
|
2006-07-24 17:50:54 +00:00
|
|
|
== -1) {
|
|
|
|
::close(socket);
|
2004-12-30 22:43:55 +00:00
|
|
|
return false;
|
2006-07-24 17:50:54 +00:00
|
|
|
}
|
2004-12-30 22:43:55 +00:00
|
|
|
|
2012-03-03 23:04:11 +01:00
|
|
|
if (TimeoutMs > 0 && ::fcntl(socket, F_SETFL, O_NONBLOCK) == -1) {
|
|
|
|
::close(socket);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2004-12-30 22:43:55 +00:00
|
|
|
m_RemoteAddr.sin_family = AF_INET;
|
|
|
|
m_RemoteAddr.sin_port = htons(Port);
|
2005-02-08 17:22:35 +00:00
|
|
|
m_RemoteAddr.sin_addr.s_addr = inet_addr(Host.c_str());
|
2012-03-03 23:04:11 +01:00
|
|
|
if (::connect(socket, (struct sockaddr*)&m_RemoteAddr, sizeof(m_RemoteAddr)) == -1) {
|
|
|
|
if (TimeoutMs > 0 && errno == EINPROGRESS) {
|
|
|
|
int so_error;
|
|
|
|
socklen_t len = sizeof(so_error);
|
|
|
|
cTBSelect select;
|
|
|
|
select.Add(socket);
|
|
|
|
if (select.Select(TimeoutMs) == -1 ||
|
|
|
|
::getsockopt(socket, SOL_SOCKET, SO_ERROR, &so_error, &len) == -1) {
|
|
|
|
::close(socket);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (so_error) {
|
|
|
|
errno = so_error;
|
|
|
|
::close(socket);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
::close(socket);
|
|
|
|
return false;
|
|
|
|
}
|
2006-07-24 17:50:54 +00:00
|
|
|
}
|
2004-12-30 22:43:55 +00:00
|
|
|
|
2009-02-13 10:39:20 +00:00
|
|
|
if (m_Type == SOCK_STREAM) {
|
|
|
|
len = sizeof(struct sockaddr_in);
|
|
|
|
if (::getpeername(socket, (struct sockaddr*)&m_RemoteAddr, &len) == -1) {
|
|
|
|
::close(socket);
|
|
|
|
return false;
|
|
|
|
}
|
2006-07-24 17:50:54 +00:00
|
|
|
}
|
2004-12-30 22:43:55 +00:00
|
|
|
|
|
|
|
len = sizeof(struct sockaddr_in);
|
2006-07-24 17:50:54 +00:00
|
|
|
if (::getsockname(socket, (struct sockaddr*)&m_LocalAddr, &len) == -1) {
|
|
|
|
::close(socket);
|
2004-12-30 22:43:55 +00:00
|
|
|
return false;
|
2006-07-24 17:50:54 +00:00
|
|
|
}
|
2004-12-30 22:43:55 +00:00
|
|
|
|
2009-02-13 10:39:20 +00:00
|
|
|
if (!cTBSource::Open(socket)) {
|
|
|
|
::close(socket);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
2004-12-30 22:43:55 +00:00
|
|
|
}
|
|
|
|
|
2005-02-08 17:22:35 +00:00
|
|
|
bool cTBSocket::Listen(const std::string &Ip, unsigned int Port, int BackLog) {
|
2004-12-30 22:43:55 +00:00
|
|
|
int val;
|
|
|
|
socklen_t len;
|
|
|
|
int socket;
|
|
|
|
|
|
|
|
if (IsOpen()) Close();
|
|
|
|
|
2009-02-13 10:39:20 +00:00
|
|
|
if ((socket = ::socket(PF_INET, m_Type, m_Protocol)) == -1)
|
2004-12-30 22:43:55 +00:00
|
|
|
return false;
|
|
|
|
|
|
|
|
val = 1;
|
|
|
|
if (::setsockopt(socket, SOL_SOCKET, SO_REUSEADDR, &val, sizeof(val)) == -1)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
m_LocalAddr.sin_family = AF_INET;
|
|
|
|
m_LocalAddr.sin_port = htons(Port);
|
2005-02-08 17:22:35 +00:00
|
|
|
m_LocalAddr.sin_addr.s_addr = inet_addr(Ip.c_str());
|
2004-12-30 22:43:55 +00:00
|
|
|
if (::bind(socket, (struct sockaddr*)&m_LocalAddr, sizeof(m_LocalAddr))
|
|
|
|
== -1)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
len = sizeof(struct sockaddr_in);
|
|
|
|
if (::getsockname(socket, (struct sockaddr*)&m_LocalAddr, &len) == -1)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (m_Type == SOCK_STREAM && ::listen(socket, BackLog) == -1)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (!cTBSource::Open(socket))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool cTBSocket::Accept(const cTBSocket &Listener) {
|
|
|
|
socklen_t addrlen;
|
|
|
|
int socket;
|
|
|
|
|
|
|
|
if (IsOpen()) Close();
|
|
|
|
|
|
|
|
addrlen = sizeof(struct sockaddr_in);
|
|
|
|
if ((socket = ::accept(Listener, (struct sockaddr*)&m_RemoteAddr,
|
|
|
|
&addrlen)) == -1)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
addrlen = sizeof(struct sockaddr_in);
|
|
|
|
if (::getsockname(socket, (struct sockaddr*)&m_LocalAddr, &addrlen) == -1)
|
|
|
|
return false;
|
|
|
|
|
2009-09-04 13:24:30 +00:00
|
|
|
int sol=1;
|
|
|
|
// Ignore possible errors here, proceed as usual
|
|
|
|
::setsockopt(socket, SOL_SOCKET, SO_KEEPALIVE, &sol, sizeof(sol));
|
|
|
|
|
2004-12-30 22:43:55 +00:00
|
|
|
if (!cTBSource::Open(socket))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
RETURNS(cTBSocket, cTBSocket::Accept(void) const, ret)
|
|
|
|
ret.Accept(*this);
|
|
|
|
RETURN(ret)
|
|
|
|
|
|
|
|
bool cTBSocket::Close(void) {
|
|
|
|
bool ret = true;
|
|
|
|
|
|
|
|
if (!IsOpen())
|
|
|
|
ERRNUL(EBADF);
|
|
|
|
|
|
|
|
if (::close(*this) == -1)
|
|
|
|
ret = false;
|
|
|
|
|
|
|
|
if (!cTBSource::Close())
|
|
|
|
ret = false;
|
|
|
|
|
|
|
|
memset(&m_LocalAddr, 0, sizeof(m_LocalAddr));
|
|
|
|
memset(&m_RemoteAddr, 0, sizeof(m_RemoteAddr));
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool cTBSocket::Shutdown(int how) {
|
|
|
|
if (!IsOpen())
|
|
|
|
ERRNUL(EBADF);
|
|
|
|
|
|
|
|
return ::shutdown(*this, how) != -1;
|
|
|
|
}
|
2007-05-09 09:12:42 +00:00
|
|
|
|
|
|
|
bool cTBSocket::SetDSCP(void) {
|
|
|
|
int dscp = STREAMDEV_DSCP;
|
2008-03-12 09:36:27 +00:00
|
|
|
return ::setsockopt(*this, IPPROTO_IP, IP_TOS, &dscp, sizeof(dscp)) != -1;
|
2007-05-09 09:12:42 +00:00
|
|
|
}
|