SocketDescriptor.hxx 7.15 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38
/*
 * Copyright (C) 2012-2017 Max Kellermann <max.kellermann@gmail.com>
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 *
 * - Redistributions of source code must retain the above copyright
 * notice, this list of conditions and the following disclaimer.
 *
 * - Redistributions in binary form must reproduce the above copyright
 * notice, this list of conditions and the following disclaimer in the
 * documentation and/or other materials provided with the
 * distribution.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
 * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE
 * FOUNDATION OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
 * OF THE POSSIBILITY OF SUCH DAMAGE.
 */

#ifndef SOCKET_DESCRIPTOR_HXX
#define SOCKET_DESCRIPTOR_HXX

#include "system/FileDescriptor.hxx"

#include <type_traits>

class SocketAddress;
class StaticSocketAddress;
39 40
class IPv4Address;
class IPv6Address;
41 42 43 44 45 46

/**
 * An OO wrapper for a UNIX socket descriptor.
 */
class SocketDescriptor : protected FileDescriptor {
protected:
47
	explicit constexpr SocketDescriptor(FileDescriptor _fd) noexcept
48 49 50 51 52
		:FileDescriptor(_fd) {}

public:
	SocketDescriptor() = default;

53
	explicit constexpr SocketDescriptor(int _fd) noexcept
54 55
		:FileDescriptor(_fd) {}

56
	constexpr bool operator==(SocketDescriptor other) const noexcept {
57 58 59 60 61 62 63 64 65 66
		return fd == other.fd;
	}

#ifndef _WIN32
	/**
	 * Convert a #FileDescriptor to a #SocketDescriptor.  This is only
	 * possible on operating systems where socket descriptors are the
	 * same as file descriptors (i.e. not on Windows).  Use this only
	 * when you know what you're doing.
	 */
67
	static constexpr SocketDescriptor FromFileDescriptor(FileDescriptor fd) noexcept {
68 69 70 71 72 73 74 75 76
		return SocketDescriptor(fd);
	}

	/**
	 * Convert this object to a #FileDescriptor instance.  This is only
	 * possible on operating systems where socket descriptors are the
	 * same as file descriptors (i.e. not on Windows).  Use this only
	 * when you know what you're doing.
	 */
77
	constexpr const FileDescriptor &ToFileDescriptor() const noexcept {
78 79 80 81 82
		return *this;
	}
#endif

	using FileDescriptor::IsDefined;
83
#ifndef _WIN32
84
	using FileDescriptor::IsValid;
85
	using FileDescriptor::IsSocket;
86
#endif
87 88 89 90 91 92 93 94 95 96 97 98 99 100

	/**
	 * Determine the socket type, i.e. SOCK_STREAM, SOCK_DGRAM or
	 * SOCK_SEQPACKET.  Returns -1 on error.
	 */
	gcc_pure
	int GetType() const noexcept;

	/**
	 * Is this a stream socket?
	 */
	gcc_pure
	bool IsStream() const noexcept;

101 102 103 104 105
	using FileDescriptor::Get;
	using FileDescriptor::Set;
	using FileDescriptor::Steal;
	using FileDescriptor::SetUndefined;

106
	static constexpr SocketDescriptor Undefined() noexcept {
107 108 109
		return SocketDescriptor(FileDescriptor::Undefined());
	}

110 111 112
	using FileDescriptor::EnableCloseOnExec;
	using FileDescriptor::DisableCloseOnExec;

113 114 115 116 117 118
#ifndef _WIN32
	using FileDescriptor::SetNonBlocking;
	using FileDescriptor::SetBlocking;
	using FileDescriptor::Duplicate;
	using FileDescriptor::Close;
#else
119 120
	bool SetNonBlocking() noexcept;

121 122 123 124 125 126
	/**
	 * This method replaces FileDescriptor::Close(), using closesocket()
	 * on Windows.  FileDescriptor::Close() is not virtual, so be
	 * careful when dealing with a FileDescriptor reference that is
	 * really a SocketDescriptor.
	 */
127
	void Close() noexcept;
128 129 130 131 132 133 134 135 136 137 138
#endif

	/**
	 * Create a socket.
	 *
	 * @param domain is the address domain
	 * @param type is the sochet type
	 * @param protocol is the protocol
	 * @return True on success, False on failure
	 * See man 2 socket for detailed information
	 */
139
	bool Create(int domain, int type, int protocol) noexcept;
140 141 142 143

	/**
	 * Like Create(), but enable non-blocking mode.
	 */
144
	bool CreateNonBlock(int domain, int type, int protocol) noexcept;
145

146
#ifndef _WIN32
147
	static bool CreateSocketPair(int domain, int type, int protocol,
148 149
				     SocketDescriptor &a,
				     SocketDescriptor &b) noexcept;
150
	static bool CreateSocketPairNonBlock(int domain, int type, int protocol,
151 152
					     SocketDescriptor &a,
					     SocketDescriptor &b) noexcept;
153
#endif
154

155
	int GetError() noexcept;
156

157 158 159
	/**
	 * @return the value size or 0 on error
	 */
160 161
	size_t GetOption(int level, int name,
			 void *value, size_t size) const noexcept;
162 163 164 165 166 167 168 169 170 171

#ifdef HAVE_STRUCT_UCRED
	/**
	 * Receive peer credentials (SO_PEERCRED).  On error, the pid
	 * is -1.
	 */
	gcc_pure
	struct ucred GetPeerCredentials() const noexcept;
#endif

172 173
	bool SetOption(int level, int name,
		       const void *value, size_t size) noexcept;
174

175
	bool SetBoolOption(int level, int name, bool _value) noexcept {
176 177 178 179
		const int value = _value;
		return SetOption(level, name, &value, sizeof(value));
	}

180 181
	bool SetKeepAlive(bool value=true) noexcept;
	bool SetReuseAddress(bool value=true) noexcept;
182 183

#ifdef __linux__
184 185 186 187
	bool SetReusePort(bool value=true) noexcept;
	bool SetFreeBind(bool value=true) noexcept;
	bool SetNoDelay(bool value=true) noexcept;
	bool SetCork(bool value=true) noexcept;
188

189 190
	bool SetTcpDeferAccept(const int &seconds) noexcept;
	bool SetV6Only(bool value) noexcept;
191 192 193 194

	/**
	 * Setter for SO_BINDTODEVICE.
	 */
195
	bool SetBindToDevice(const char *name) noexcept;
196

197
	bool SetTcpFastOpen(int qlen=16) noexcept;
198 199 200 201

	bool AddMembership(const IPv4Address &address) noexcept;
	bool AddMembership(const IPv6Address &address) noexcept;
	bool AddMembership(SocketAddress address) noexcept;
202 203
#endif

204
	bool Bind(SocketAddress address) noexcept;
205 206 207 208 209

#ifdef __linux__
	/**
	 * Binds the socket to a unique abstract address.
	 */
210
	bool AutoBind() noexcept;
211 212
#endif

213
	bool Listen(int backlog) noexcept;
214

215 216 217
	SocketDescriptor Accept() noexcept;
	SocketDescriptor AcceptNonBlock() const noexcept;
	SocketDescriptor AcceptNonBlock(StaticSocketAddress &address) const noexcept;
218

219
	bool Connect(SocketAddress address) noexcept;
220 221

	gcc_pure
222
	StaticSocketAddress GetLocalAddress() const noexcept;
223 224

	gcc_pure
225
	StaticSocketAddress GetPeerAddress() const noexcept;
226

227 228
	ssize_t Read(void *buffer, size_t length) noexcept;
	ssize_t Write(const void *buffer, size_t length) noexcept;
229 230

#ifdef _WIN32
231 232
	int WaitReadable(int timeout_ms) const noexcept;
	int WaitWritable(int timeout_ms) const noexcept;
233 234 235 236 237 238 239 240 241 242
#else
	using FileDescriptor::WaitReadable;
	using FileDescriptor::WaitWritable;
	using FileDescriptor::IsReadyForWriting;
#endif

	/**
	 * Receive a datagram and return the source address.
	 */
	ssize_t Read(void *buffer, size_t length,
243
		     StaticSocketAddress &address) noexcept;
244 245 246 247 248

	/**
	 * Send a datagram to the specified address.
	 */
	ssize_t Write(const void *buffer, size_t length,
249
		      SocketAddress address) noexcept;
250 251 252 253 254 255

#ifndef _WIN32
	void Shutdown() noexcept;
	void ShutdownRead() noexcept;
	void ShutdownWrite() noexcept;
#endif
256 257 258 259 260
};

static_assert(std::is_trivial<SocketDescriptor>::value, "type is not trivial");

#endif