/* * IXSocket.h * Author: Benjamin Sergeant * Copyright (c) 2017-2018 Machine Zone, Inc. All rights reserved. */ #pragma once #include #include #include #include #include #include #ifdef _WIN32 #include typedef SSIZE_T ssize_t; #undef EWOULDBLOCK #undef EAGAIN #undef EINPROGRESS #undef EBADF #undef EINVAL // map to WSA error codes #define EWOULDBLOCK WSAEWOULDBLOCK #define EAGAIN WSATRY_AGAIN #define EINPROGRESS WSAEINPROGRESS #define EBADF WSAEBADF #define EINVAL WSAEINVAL #endif #include "IXCancellationRequest.h" #include "IXProgressCallback.h" namespace ix { class SelectInterrupt; using SelectInterruptPtr = std::unique_ptr; enum class PollResultType { ReadyForRead = 0, ReadyForWrite = 1, Timeout = 2, Error = 3, SendRequest = 4, CloseRequest = 5 }; class Socket { public: Socket(int fd = -1); virtual ~Socket(); bool init(std::string& errorMsg); // Functions to check whether there is activity on the socket PollResultType poll(int timeoutMs = kDefaultPollTimeout); bool wakeUpFromPoll(uint64_t wakeUpCode); PollResultType isReadyToWrite(int timeoutMs); PollResultType isReadyToRead(int timeoutMs); // Virtual methods virtual bool accept(std::string& errMsg); virtual bool connect(const std::string& host, int port, std::string& errMsg, const CancellationRequest& isCancellationRequested); virtual void close(); virtual ssize_t send(char* buffer, size_t length); ssize_t send(const std::string& buffer); virtual ssize_t recv(void* buffer, size_t length); // Blocking and cancellable versions, working with socket that can be set // to non blocking mode. Used during HTTP upgrade. bool readByte(void* buffer, const CancellationRequest& isCancellationRequested); bool writeBytes(const std::string& str, const CancellationRequest& isCancellationRequested); std::pair readLine(const CancellationRequest& isCancellationRequested); std::pair readBytes(size_t length, const OnProgressCallback& onProgressCallback, const CancellationRequest& isCancellationRequested); static int getErrno(); static bool isWaitNeeded(); static void closeSocket(int fd); static PollResultType poll(bool readyToRead, int timeoutMs, int sockfd, const SelectInterruptPtr& selectInterrupt); // Used as special codes for pipe communication static const uint64_t kSendRequest; static const uint64_t kCloseRequest; protected: std::atomic _sockfd; std::mutex _socketMutex; private: static const int kDefaultPollTimeout; static const int kDefaultPollNoTimeout; // Buffer for reading from our socket. That buffer is never resized. std::vector _readBuffer; static constexpr size_t kChunkSize = 1 << 15; SelectInterruptPtr _selectInterrupt; }; } // namespace ix