131 lines
3.6 KiB
C++
131 lines
3.6 KiB
C++
/*
|
|
* IXSocket.h
|
|
* Author: Benjamin Sergeant
|
|
* Copyright (c) 2017-2018 Machine Zone, Inc. All rights reserved.
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include <atomic>
|
|
#include <functional>
|
|
#include <memory>
|
|
#include <mutex>
|
|
#include <string>
|
|
#include <vector>
|
|
|
|
// For kqueue
|
|
#if defined(__APPLE__)
|
|
#include <sys/types.h>
|
|
#include <sys/event.h>
|
|
#include <sys/time.h>
|
|
#endif
|
|
|
|
#ifdef _WIN32
|
|
#include <BaseTsd.h>
|
|
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<SelectInterrupt>;
|
|
|
|
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<bool, std::string> readLine(const CancellationRequest& isCancellationRequested);
|
|
std::pair<bool, std::string> 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,
|
|
int kqueuefd);
|
|
|
|
|
|
// Used as special codes for pipe communication
|
|
static const uint64_t kSendRequest;
|
|
static const uint64_t kCloseRequest;
|
|
|
|
protected:
|
|
std::atomic<int> _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<uint8_t> _readBuffer;
|
|
static constexpr size_t kChunkSize = 1 << 15;
|
|
|
|
SelectInterruptPtr _selectInterrupt;
|
|
|
|
#if defined(__APPLE__)
|
|
int _kqueuefd;
|
|
#endif
|
|
};
|
|
} // namespace ix
|