blob: 90298f6bcab2bf10b9f352bf5846aa30fc0f926b [file] [log] [blame]
jat@google.com134be542009-08-03 15:30:11 +00001#ifndef _H_Socket
2#define _H_Socket
3/*
4 * Copyright 2008 Google Inc.
5 *
6 * Licensed under the Apache License, Version 2.0 (the "License"); you may not
7 * use this file except in compliance with the License. You may obtain a copy of
8 * the License at
9 *
10 * http://www.apache.org/licenses/LICENSE-2.0
11 *
12 * Unless required by applicable law or agreed to in writing, software
13 * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
14 * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
15 * License for the specific language governing permissions and limitations under
16 * the License.
17 */
18
19#include "Platform.h"
20#include "Debug.h"
21
22#include <string>
23
24#ifdef _WINDOWS
25#include <winsock2.h>
26#include <ws2tcpip.h>
27#else
28#include <netdb.h>
29#include <sys/socket.h>
30#include <netinet/in.h>
31#include <arpa/inet.h>
32#include <unistd.h>
33#include <sys/time.h>
34#endif
35
36/**
37 * Encapsulates platform dependencies regarding buffered sockets.
38 */
39class Socket {
40private:
41 // Buffer size, chosen to fit in a single packet after TCP/IP overhead.
42 static const int BUF_SIZE = 1400;
43
44 // Can't rely on a sentinel value for the socket descriptor
45 bool connected;
46
47 SOCKETTYPE sock;
48
49 // Read buffer
50 char* readBuf;
51
52 // One bye past end of valid data in readBuf
53 char* readValid;
54
55 // Current read pointer
56 char* readBufPtr;
57
58 // Write buffer
59 char* writeBuf;
60
61 // Current write pointer
62 char* writeBufPtr;
63
64 // Stats
65 unsigned long numReads;
66 unsigned long long totReadBytes;
67 size_t maxReadBytes;
68
69 unsigned long numWrites;
70 unsigned long long totWriteBytes;
71 size_t maxWriteBytes;
72
73private:
74 void init();
75 bool fillReadBuf();
76 bool emptyWriteBuf();
77
78public:
79 Socket() : connected(false), readBuf(new char[BUF_SIZE]), writeBuf(new char[BUF_SIZE]) {
80 readBufPtr = readValid = readBuf;
81 writeBufPtr = writeBuf;
82 numReads = numWrites = 0;
83 maxReadBytes = maxWriteBytes = 0;
84 totReadBytes = totWriteBytes = 0;
85 init();
86 }
87
88 ~Socket() {
89 disconnect();
90#ifdef _WINDOWS
91 if (0) WSACleanup();
92#endif
93 // TODO(jat): LEAK LEAK LEAK
94 // delete[] readBuf;
95 // delete[] writeBuf;
96 Debug::log(Debug::Debugging) << "Socket: #r=" << numReads << ", bytes/read="
97 << (numReads ? totReadBytes / numReads : 0) << ", maxr=" << maxReadBytes << "; #w="
98 << numWrites << ", bytes/write=" << (numWrites ? totWriteBytes / numWrites : 0) << ", maxw="
99 << maxWriteBytes << Debug::flush;
100 }
101
102 /**
103 * Connects this socket to a specified port on a host.
104 *
105 * @param host host name or IP address to connect to
106 * @param port TCP port to connect to
107 * @return true if the connection succeeds
108 */
109 bool connect(const char* host, int port);
110
111 /**
112 * Returns true if the socket is connected.
113 */
114 bool isConnected() const {
115 return connected;
116 }
117
118 /**
119 * Disconnect this socket.
120 *
121 * @param doFlush true (the default value) if the socket should be flushed.
122 * @return true if disconnect succeeds
123 */
124 bool disconnect(bool doFlush = true);
125
126 /**
127 * Read a single byte from the socket.
128 *
129 * @return -1 on error, otherwise unsigned byte read.
130 */
131 int readByte() {
132 if (readBufPtr >= readValid) {
133 if (!fillReadBuf()) {
134 return -1;
135 }
136 }
137 return static_cast<unsigned char>(*readBufPtr++);
138 }
139
140 /**
141 * Write a single byte to the socket.
142 *
143 * @return true on success.
144 */
145 bool writeByte(char c) {
146 if (writeBufPtr >= writeBuf + BUF_SIZE) {
147 if (!emptyWriteBuf()) {
148 return false;
149 }
150 }
151 *writeBufPtr++ = c;
152 return true;
153 }
154
155 /**
156 * Flush any pending writes on the socket.
157 *
158 * @return true on success
159 */
160 bool flush() {
161 if (writeBufPtr > writeBuf) {
162 if (!emptyWriteBuf()) {
163 return false;
164 }
165 }
166 return true;
167 }
168};
169
170#endif