Merge Phy<> from netcon.

This commit is contained in:
Adam Ierymenko 2015-10-27 12:50:23 -07:00
parent cfe166ef35
commit 7295fcfa86
4 changed files with 65 additions and 74 deletions

View File

@ -100,6 +100,14 @@ struct HttpPhyHandler
phy->setNotifyWritable(sock,false); phy->setNotifyWritable(sock,false);
} }
inline void phyOnFileDescriptorActivity(PhySocket *sock,void **uptr,bool readable,bool writable) {}
#ifdef __UNIX_LIKE__
inline void phyOnUnixAccept(PhySocket *sockL,PhySocket *sockN,void **uptrL,void **uptrN) {}
inline void phyOnUnixClose(PhySocket *sock,void **uptr) {}
inline void phyOnUnixData(PhySocket *sock,void **uptr,void *data,unsigned long len) {}
inline void phyOnUnixWritable(PhySocket *sock,void **uptr) {}
#endif // __UNIX_LIKE__
http_parser parser; http_parser parser;
std::string currentHeaderField; std::string currentHeaderField;
std::string currentHeaderValue; std::string currentHeaderValue;

View File

@ -78,11 +78,6 @@
#define ZT_PHY_MAX_INTERCEPTS ZT_PHY_MAX_SOCKETS #define ZT_PHY_MAX_INTERCEPTS ZT_PHY_MAX_SOCKETS
#define ZT_PHY_SOCKADDR_STORAGE_TYPE struct sockaddr_storage #define ZT_PHY_SOCKADDR_STORAGE_TYPE struct sockaddr_storage
#if defined(__linux__) || defined(linux) || defined(__LINUX__) || defined(__linux)
#define ZT_PHY_HAVE_EVENTFD 1
#include <sys/eventfd.h>
#endif
#endif // Windows or not #endif // Windows or not
namespace ZeroTier { namespace ZeroTier {
@ -109,6 +104,7 @@ typedef void PhySocket;
* phyOnTcpClose(PhySocket *sock,void **uptr) * phyOnTcpClose(PhySocket *sock,void **uptr)
* phyOnTcpData(PhySocket *sock,void **uptr,void *data,unsigned long len) * phyOnTcpData(PhySocket *sock,void **uptr,void *data,unsigned long len)
* phyOnTcpWritable(PhySocket *sock,void **uptr) * phyOnTcpWritable(PhySocket *sock,void **uptr)
* phyOnFileDescriptorActivity(PhySocket *sock,void **uptr,bool readable,bool writable)
* *
* On Linux/OSX/Unix only (not required/used on Windows or elsewhere): * On Linux/OSX/Unix only (not required/used on Windows or elsewhere):
* *
@ -116,9 +112,6 @@ typedef void PhySocket;
* phyOnUnixClose(PhySocket *sock,void **uptr) * phyOnUnixClose(PhySocket *sock,void **uptr)
* phyOnUnixData(PhySocket *sock,void **uptr,void *data,unsigned long len) * phyOnUnixData(PhySocket *sock,void **uptr,void *data,unsigned long len)
* phyOnUnixWritable(PhySocket *sock,void **uptr) * phyOnUnixWritable(PhySocket *sock,void **uptr)
* phyOnSocketPairEndpointClose(PhySocket *sock,void **uptr)
* phyOnSocketPairEndpointData(PhySocket *sock,void **uptr,void *data,unsigned long len)
* phyOnSocketPairEndpointWritable(PhySocket *sock,void **uptr)
* *
* These templates typically refer to function objects. Templates are used to * These templates typically refer to function objects. Templates are used to
* avoid the call overhead of indirection, which is surprisingly high for high * avoid the call overhead of indirection, which is surprisingly high for high
@ -154,11 +147,10 @@ private:
ZT_PHY_SOCKET_TCP_OUT_CONNECTED = 0x02, ZT_PHY_SOCKET_TCP_OUT_CONNECTED = 0x02,
ZT_PHY_SOCKET_TCP_IN = 0x03, ZT_PHY_SOCKET_TCP_IN = 0x03,
ZT_PHY_SOCKET_TCP_LISTEN = 0x04, ZT_PHY_SOCKET_TCP_LISTEN = 0x04,
ZT_PHY_SOCKET_RAW = 0x05, ZT_PHY_SOCKET_UDP = 0x05,
ZT_PHY_SOCKET_UDP = 0x06, ZT_PHY_SOCKET_FD = 0x06,
ZT_PHY_SOCKET_UNIX_IN = 0x07, ZT_PHY_SOCKET_UNIX_IN = 0x07,
ZT_PHY_SOCKET_UNIX_LISTEN = 0x08, ZT_PHY_SOCKET_UNIX_LISTEN = 0x08
ZT_PHY_SOCKET_PAIR_ENDPOINT = 0x09
}; };
struct PhySocketImpl struct PhySocketImpl
@ -277,57 +269,47 @@ public:
*/ */
inline unsigned long maxCount() const throw() { return ZT_PHY_MAX_SOCKETS; } inline unsigned long maxCount() const throw() { return ZT_PHY_MAX_SOCKETS; }
#ifdef __UNIX_LIKE__
/** /**
* Create a two-way socket pair * Wrap a raw file descriptor in a PhySocket structure
* *
* This uses socketpair() to create a local domain pair. The returned * This can be used to select/poll on a raw file descriptor as part of this
* PhySocket holds the local side of the socket pair, while the * class's I/O loop. By default the fd is set for read notification but
* supplied fd variable is set to the descriptor for the remote side. * this can be controlled with setNotifyReadable(). When any detected
* condition is present, the phyOnFileDescriptorActivity() callback is
* called with one or both of its arguments 'true'.
* *
* The local side is set to O_NONBLOCK to work with our poll loop, but * The Phy<>::close() method *must* be called when you're done with this
* the remote descriptor is left untouched. It's up to the caller to * file descriptor to remove it from the select/poll set, but unlike other
* set any required fcntl(), ioctl(), or setsockopt() settings there. * types of sockets Phy<> does not actually close the underlying fd or
* It's also up to the caller to close the remote descriptor when * otherwise manage its life cycle. There is also no close notification
* done, if necessary. * callback for this fd, since Phy<> doesn't actually perform reading or
* writing or detect error conditions. This is only useful for adding a
* file descriptor to Phy<> to select/poll on it.
* *
* @param remoteSocketDescriptor Result parameter set to remote end of socket pair's socket FD * @param fd Raw file descriptor
* @param uptr Pointer to associate with local side of socket pair * @param uptr User pointer to supply to callbacks
* @return PhySocket for local side of socket pair * @return PhySocket wrapping fd or NULL on failure (out of memory or too many sockets)
*/ */
inline PhySocket *createSocketPair(ZT_PHY_SOCKFD_TYPE &remoteSocketDescriptor,void *uptr = (void *)0) inline PhySocket *wrapSocket(ZT_PHY_SOCKFD_TYPE fd,void *uptr = (void *)0)
{ {
if (_socks.size() >= ZT_PHY_MAX_SOCKETS) if (_socks.size() >= ZT_PHY_MAX_SOCKETS)
return (PhySocket *)0; return (PhySocket *)0;
int fd[2]; fd[0] = -1; fd[1] = -1;
if ((::socketpair(PF_LOCAL,SOCK_STREAM,0,fd) != 0)||(fd[0] <= 0)||(fd[1] <= 0))
return (PhySocket *)0;
fcntl(fd[0],F_SETFL,O_NONBLOCK);
try { try {
_socks.push_back(PhySocketImpl()); _socks.push_back(PhySocketImpl());
} catch ( ... ) { } catch ( ... ) {
ZT_PHY_CLOSE_SOCKET(fd[0]);
ZT_PHY_CLOSE_SOCKET(fd[1]);
return (PhySocket *)0; return (PhySocket *)0;
} }
PhySocketImpl &sws = _socks.back(); PhySocketImpl &sws = _socks.back();
if ((long)fd > _nfds)
if ((long)fd[0] > _nfds) _nfds = (long)fd;
_nfds = (long)fd[0]; FD_SET(fd,&_readfds);
FD_SET(fd[0],&_readfds); sws.type = ZT_PHY_SOCKET_FD;
sws.type = ZT_PHY_SOCKET_PAIR_ENDPOINT; sws.sock = fd;
sws.sock = fd[0];
sws.uptr = uptr; sws.uptr = uptr;
memset(&(sws.saddr),0,sizeof(struct sockaddr_storage)); memset(&(sws.saddr),0,sizeof(struct sockaddr_storage));
// no sockaddr for this socket type, leave saddr null // no sockaddr for this socket type, leave saddr null
remoteSocketDescriptor = fd[1];
return (PhySocket *)&sws; return (PhySocket *)&sws;
} }
#endif // __UNIX_LIKE__
/** /**
* Bind a UDP socket * Bind a UDP socket
@ -787,6 +769,26 @@ public:
} }
} }
/**
* Set whether we want to be notified that a socket is readable
*
* This is primarily for raw sockets added with wrapSocket(). It could be
* used with others, but doing so would essentially lock them and prevent
* data from being read from them until this is set to 'true' again.
*
* @param sock Socket to modify
* @param notifyReadable True if socket should be monitored for readability
*/
inline const void setNotifyReadable(PhySocket *sock,bool notifyReadable)
{
PhySocketImpl &sws = *(reinterpret_cast<PhySocketImpl *>(sock));
if (notifyReadable) {
FD_SET(sws.sock,&_readfds);
} else {
FD_CLR(sws.sock,&_readfds);
}
}
/** /**
* Wait for activity and handle one or more events * Wait for activity and handle one or more events
* *
@ -936,7 +938,7 @@ public:
} }
if ((FD_ISSET(sock,&wfds))&&(FD_ISSET(sock,&_writefds))) { if ((FD_ISSET(sock,&wfds))&&(FD_ISSET(sock,&_writefds))) {
try { try {
_handler->phyOnUnixWritable((PhySocket *)&(*s),&(s->uptr)); //_handler->phyOnUnixWritable((PhySocket *)&(*s),&(s->uptr));
} catch ( ... ) {} } catch ( ... ) {}
} }
#endif // __UNIX_LIKE__ #endif // __UNIX_LIKE__
@ -971,25 +973,15 @@ public:
#endif // __UNIX_LIKE__ #endif // __UNIX_LIKE__
break; break;
case ZT_PHY_SOCKET_PAIR_ENDPOINT: { case ZT_PHY_SOCKET_FD: {
#ifdef __UNIX_LIKE__ ZT_PHY_SOCKFD_TYPE sock = s->sock;
ZT_PHY_SOCKFD_TYPE sock = s->sock; // if closed, s->sock becomes invalid as s is no longer dereferencable const bool readable = ((FD_ISSET(sock,&rfds))&&(FD_ISSET(sock,&_readfds)));
if (FD_ISSET(sock,&rfds)) { const bool writable = ((FD_ISSET(sock,&wfds))&&(FD_ISSET(sock,&_writefds)));
long n = (long)::read(sock,buf,sizeof(buf)); if ((readable)||(writable)) {
if (n <= 0) {
this->close((PhySocket *)&(*s),true);
} else {
try { try {
_handler->phyOnSocketPairEndpointData((PhySocket *)&(*s),&(s->uptr),(void *)buf,(unsigned long)n); _handler->phyOnFileDescriptorActivity((PhySocket *)&(*s),&(s->uptr),readable,writable);
} catch ( ... ) {} } catch ( ... ) {}
} }
}
if ((FD_ISSET(sock,&wfds))&&(FD_ISSET(sock,&_writefds))) {
try {
_handler->phyOnSocketPairEndpointWritable((PhySocket *)&(*s),&(s->uptr));
} catch ( ... ) {}
}
#endif // __UNIX_LIKE__
} break; } break;
default: default:
@ -1021,6 +1013,7 @@ public:
FD_CLR(sws.sock,&_exceptfds); FD_CLR(sws.sock,&_exceptfds);
#endif #endif
if (sws.type != ZT_PHY_SOCKET_FD)
ZT_PHY_CLOSE_SOCKET(sws.sock); ZT_PHY_CLOSE_SOCKET(sws.sock);
#ifdef __UNIX_LIKE__ #ifdef __UNIX_LIKE__
@ -1048,12 +1041,6 @@ public:
} catch ( ... ) {} } catch ( ... ) {}
#endif // __UNIX_LIKE__ #endif // __UNIX_LIKE__
break; break;
case ZT_PHY_SOCKET_PAIR_ENDPOINT:
#ifdef __UNIX_LIKE__
try {
_handler->phyOnSocketPairEndpointClose(sock,&(sws.uptr));
} catch ( ... ) {}
#endif // __UNIX_LIKE__
default: default:
break; break;
} }

View File

@ -837,9 +837,7 @@ struct TestPhyHandlers
inline void phyOnUnixClose(PhySocket *sock,void **uptr) {} inline void phyOnUnixClose(PhySocket *sock,void **uptr) {}
inline void phyOnUnixData(PhySocket *sock,void **uptr,void *data,unsigned long len) {} inline void phyOnUnixData(PhySocket *sock,void **uptr,void *data,unsigned long len) {}
inline void phyOnUnixWritable(PhySocket *sock,void **uptr) {} inline void phyOnUnixWritable(PhySocket *sock,void **uptr) {}
inline void phyOnSocketPairEndpointClose(PhySocket *sock,void **uptr) {} inline void phyOnFileDescriptorActivity(PhySocket *sock,void **uptr,bool readable,bool writable) {}
inline void phyOnSocketPairEndpointData(PhySocket *sock,void **uptr,void *data,unsigned long len) {}
inline void phyOnSocketPairEndpointWritable(PhySocket *sock,void **uptr) {}
#endif // __UNIX_LIKE__ #endif // __UNIX_LIKE__
}; };
static int testPhy() static int testPhy()

View File

@ -1085,9 +1085,7 @@ public:
inline void phyOnUnixClose(PhySocket *sock,void **uptr) {} inline void phyOnUnixClose(PhySocket *sock,void **uptr) {}
inline void phyOnUnixData(PhySocket *sock,void **uptr,void *data,unsigned long len) {} inline void phyOnUnixData(PhySocket *sock,void **uptr,void *data,unsigned long len) {}
inline void phyOnUnixWritable(PhySocket *sock,void **uptr) {} inline void phyOnUnixWritable(PhySocket *sock,void **uptr) {}
inline void phyOnSocketPairEndpointClose(PhySocket *sock,void **uptr) {} inline void phyOnFileDescriptorActivity(PhySocket *sock,void **uptr,bool readable,bool writable) {}
inline void phyOnSocketPairEndpointData(PhySocket *sock,void **uptr,void *data,unsigned long len) {}
inline void phyOnSocketPairEndpointWritable(PhySocket *sock,void **uptr) {}
inline int nodeVirtualNetworkConfigFunction(uint64_t nwid,enum ZT_VirtualNetworkConfigOperation op,const ZT_VirtualNetworkConfig *nwc) inline int nodeVirtualNetworkConfigFunction(uint64_t nwid,enum ZT_VirtualNetworkConfigOperation op,const ZT_VirtualNetworkConfig *nwc)
{ {