/* * * (C) 2003-2012 Anope Team * Contact us at team@anope.org * * Please read COPYING and README for further details. * * Based on the original code of Epona by Lara. * Based on the original code of Services by Andy Church. */ #ifndef SOCKETS_H #define SOCKETS_H #ifndef _WIN32 #include #include #include #endif #include "anope.h" #define NET_BUFSIZE 65535 /** A sockaddr union used to combine IPv4 and IPv6 sockaddrs */ union CoreExport sockaddrs { sockaddr sa; sockaddr_in sa4; sockaddr_in6 sa6; /** Construct the object, sets everything to 0 */ sockaddrs(const Anope::string &address = ""); /** Memset the object to 0 */ void clear(); /** Get the size of the sockaddr we represent * @return The size */ size_t size() const; /** Get the port represented by this addr * @return The port, or -1 on fail */ int port() const; /** Get the address represented by this addr * @return The address */ Anope::string addr() const; /** Check if this sockaddr has data in it */ bool operator()() const; /** Compares with sockaddr with another. Compares address type, port, and address * @return true if they are the same */ bool operator==(const sockaddrs &other) const; /* The same as above but not */ inline bool operator!=(const sockaddrs &other) const { return !(*this == other); } /** The equivalent of inet_pton * @param type AF_INET or AF_INET6 * @param address The address to place in the sockaddr structures * @param pport An option port to include in the sockaddr structures * @throws A socket exception if given invalid IPs */ void pton(int type, const Anope::string &address, int pport = 0); /** The equivalent of inet_ntop * @param type AF_INET or AF_INET6 * @param address The in_addr or in_addr6 structure * @throws A socket exception if given an invalid structure */ void ntop(int type, const void *src); }; class CoreExport cidr { sockaddrs addr; Anope::string cidr_ip; unsigned char cidr_len; public: cidr(const Anope::string &ip); cidr(const Anope::string &ip, unsigned char len); Anope::string mask() const; bool match(sockaddrs &other); }; class SocketException : public CoreException { public: /** Default constructor for socket exceptions * @param message Error message */ SocketException(const Anope::string &message) : CoreException(message) { } /** Default destructor * @throws Nothing */ virtual ~SocketException() throw() { } }; enum SocketFlag { SF_DEAD, SF_WRITABLE, SF_CONNECTING, SF_CONNECTED, SF_ACCEPTING, SF_ACCEPTED }; static const Anope::string SocketFlagStrings[] = { "SF_DEAD", "SF_WRITABLE", "SF_CONNECTING", "SF_CONNECTED", "SF_ACCEPTING", "SF_ACCEPTED", "" }; class CoreExport SocketIO { public: /** Receive something from the buffer * @param s The socket * @param buf The buf to read to * @param sz How much to read * @return Number of bytes received */ virtual int Recv(Socket *s, char *buf, size_t sz); /** Write something to the socket * @param s The socket * @param buf The data to write * @param size The length of the data */ virtual int Send(Socket *s, const char *buf, size_t sz); int Send(Socket *s, const Anope::string &buf); /** Accept a connection from a socket * @param s The socket * @return The new socket */ virtual ClientSocket *Accept(ListenSocket *s); /** Finished accepting a connection from a socket * @param s The socket * @return SF_ACCEPTED if accepted, SF_ACCEPTING if still in process, SF_DEAD on error */ virtual SocketFlag FinishAccept(ClientSocket *cs); /** Bind a socket * @param s The socket * @param ip The IP to bind to * @param port The optional port to bind to */ virtual void Bind(Socket *s, const Anope::string &ip, int port = 0); /** Connect the socket * @param s The socket * @param target IP to connect to * @param port to connect to */ virtual void Connect(ConnectionSocket *s, const Anope::string &target, int port); /** Called to potentially finish a pending connection * @param s The socket * @return SF_CONNECTED on success, SF_CONNECTING if still pending, and SF_DEAD on error. */ virtual SocketFlag FinishConnect(ConnectionSocket *s); /** Called when the socket is destructing */ virtual void Destroy() { } }; class CoreExport Socket : public Flags { protected: /* Socket FD */ int Sock; /* Is this an IPv6 socket? */ bool IPv6; public: /* Sockaddrs for bind() (if it's bound) */ sockaddrs bindaddr; /* I/O functions used for this socket */ SocketIO *IO; /** Empty constructor, should not be called. */ Socket(); /** Default constructor * @param sock The socket to use, -1 if we need to create our own * @param ipv6 true if using ipv6 * @param type The socket type, defaults to SOCK_STREAM */ Socket(int sock, bool ipv6 = false, int type = SOCK_STREAM); /** Default destructor */ virtual ~Socket(); /** Get the socket FD for this socket * @return the fd */ int GetFD() const; /** Check if this socket is IPv6 * @return true or false */ bool IsIPv6() const; /** Mark a socket as blockig * @return true if the socket is now blocking */ bool SetBlocking(); /** Mark a socket as non-blocking * @return true if the socket is now non-blocking */ bool SetNonBlocking(); /** Bind the socket to an ip and port * @param ip The ip * @param port The port */ void Bind(const Anope::string &ip, int port = 0); /** Called when there either is a read or write event. * @return true to continue to call ProcessRead/ProcessWrite, false to not continue */ virtual bool Process(); /** Called when there is something to be received for this socket * @return true on success, false to drop this socket */ virtual bool ProcessRead(); /** Called when the socket is ready to be written to * @return true on success, false to drop this socket */ virtual bool ProcessWrite(); /** Called when there is an error for this socket * @return true on success, false to drop this socket */ virtual void ProcessError(); }; class CoreExport BufferedSocket : public virtual Socket { protected: /* Things to be written to the socket */ Anope::string WriteBuffer; /* Part of a message sent from the server, but not totally received */ Anope::string extrabuf; /* How much data was received from this socket */ int RecvLen; public: /** Constructor */ BufferedSocket(); /** Default destructor */ virtual ~BufferedSocket(); /** Called when there is something to be received for this socket * @return true on success, false to drop this socket */ bool ProcessRead() anope_override; /** Called when the socket is ready to be written to * @return true on success, false to drop this socket */ bool ProcessWrite() anope_override; /** Called with a line received from the socket * @param buf The line * @return true to continue reading, false to drop the socket */ virtual bool Read(const Anope::string &buf); /** Write to the socket * @param message The message */ protected: virtual void Write(const char *buffer, size_t l); public: void Write(const char *message, ...); void Write(const Anope::string &message); /** Get the length of the read buffer * @return The length of the read buffer */ int ReadBufferLen() const; /** Get the length of the write buffer * @return The length of the write buffer */ int WriteBufferLen() const; }; class CoreExport BinarySocket : public virtual Socket { protected: struct DataBlock { char *orig; char *buf; size_t len; DataBlock(const char *b, size_t l); ~DataBlock(); }; std::deque WriteBuffer; public: /** Constructor */ BinarySocket(); /** Default destructor */ virtual ~BinarySocket(); /** Called when there is something to be received for this socket * @return true on success, false to drop this socket */ bool ProcessRead() anope_override; /** Called when the socket is ready to be written to * @return true on success, false to drop this socket */ bool ProcessWrite() anope_override; /** Write data to the socket * @param buffer The data to write * @param l The length of the data */ virtual void Write(const char *buffer, size_t l); void Write(const char *message, ...); void Write(const Anope::string &message); /** Called with data from the socket * @param buffer The data * @param l The length of buffer * @return true to continue reading, false to drop the socket */ virtual bool Read(const char *buffer, size_t l); }; class CoreExport ListenSocket : public Socket { public: /** Constructor * @param bindip The IP to bind to * @param port The port to listen on * @param ipv6 true for ipv6 */ ListenSocket(const Anope::string &bindip, int port, bool ipv6); /** Destructor */ virtual ~ListenSocket(); /** Process what has come in from the connection * @return false to destory this socket */ bool ProcessRead(); /** Called when a connection is accepted * @param fd The FD for the new connection * @param addr The sockaddr for where the connection came from * @return The new socket */ virtual ClientSocket *OnAccept(int fd, const sockaddrs &addr) = 0; }; class CoreExport ConnectionSocket : public virtual Socket { public: /* Sockaddrs for connection ip/port */ sockaddrs conaddr; /** Constructor */ ConnectionSocket(); /** Connect the socket * @param TargetHost The target host to connect to * @param Port The target port to connect to */ void Connect(const Anope::string &TargetHost, int Port); /** Called when there either is a read or write event. * Used to determine whether or not this socket is connected yet. * @return true to continue to call ProcessRead/ProcessWrite, false to not continue */ bool Process() anope_override; /** Called when there is an error for this socket * @return true on success, false to drop this socket */ void ProcessError() anope_override; /** Called on a successful connect */ virtual void OnConnect(); /** Called when a connection is not successful * @param error The error */ virtual void OnError(const Anope::string &error); }; class CoreExport ClientSocket : public virtual Socket { public: /* Listen socket this connection came from */ ListenSocket *LS; /* Clients address */ sockaddrs clientaddr; /** Constructor * @param ls Listen socket this connection is from * @param addr Address the connection came from */ ClientSocket(ListenSocket *ls, const sockaddrs &addr); /** Called when there either is a read or write event. * Used to determine whether or not this socket is connected yet. * @return true to continue to call ProcessRead/ProcessWrite, false to not continue */ bool Process() anope_override; /** Called when there is an error for this socket * @return true on success, false to drop this socket */ void ProcessError() anope_override; /** Called when a client has been accepted() successfully. */ virtual void OnAccept(); /** Called when there was an error accepting the client */ virtual void OnError(const Anope::string &error); }; class CoreExport Pipe : public Socket { public: /** The FD of the write pipe (if this isn't evenfd) * this->Sock is the readfd */ int WritePipe; /** Constructor */ Pipe(); /** Destructor */ ~Pipe(); /** Called when data is to be read */ bool ProcessRead() anope_override; /** Called when this pipe needs to be woken up */ void Notify(); /** Overload to do something useful */ virtual void OnNotify() = 0; }; extern CoreExport uint32_t TotalRead; extern CoreExport uint32_t TotalWritten; extern CoreExport SocketIO normalSocketIO; #endif // SOCKET_H