SocketHandle.h 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509
  1. /*
  2. ** Copyright 2003-2009, Ernest Laurentin (http://www.ernzo.com/)
  3. **
  4. ** Licensed under the Apache License, Version 2.0 (the "License");
  5. ** you may not use this file except in compliance with the License.
  6. ** You may obtain a copy of the License at
  7. **
  8. ** http://www.apache.org/licenses/LICENSE-2.0
  9. **
  10. ** Unless required by applicable law or agreed to in writing, software
  11. ** distributed under the License is distributed on an "AS IS" BASIS,
  12. ** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. ** See the License for the specific language governing permissions and
  14. ** limitations under the License.
  15. **
  16. ** File: SocketHandle.h
  17. ** Version: 1.4 - IPv6 support
  18. ** 1.3 - Update for Asynchronous mode / Linux port
  19. ** 1.2 - Update interface for TCP remote connection
  20. ** 1.1 - Added multicast support
  21. */
  22. #ifndef SOCKETHANDLE_H
  23. #define SOCKETHANDLE_H
  24. #pragma once
  25. #ifdef WIN32
  26. #include <winsock2.h>
  27. #include <ws2tcpip.h>
  28. #elif BSD_SOCKET
  29. #include "platform.h"
  30. #endif
  31. /**
  32. * @defgroup _SockHandle Socket Communication Classes
  33. * @{
  34. */
  35. /**
  36. * SockAddrIn structure
  37. * Encapsulate SOCKADDR_STORAGE (IPv4: SOCKADDR_IN, IPv6: SOCKADDR_IN6) structures
  38. */
  39. struct SockAddrIn : public sockaddr_storage {
  40. public:
  41. /**
  42. * Default constructor
  43. */
  44. SockAddrIn();
  45. /**
  46. * Copy constructor
  47. * @param sin reference object
  48. */
  49. SockAddrIn(const SockAddrIn& sin);
  50. /**
  51. * Destructor
  52. */
  53. ~SockAddrIn();
  54. /**
  55. * Copy function
  56. * @param sin reference object
  57. * @return reference of 'this' object
  58. */
  59. SockAddrIn& Copy(const SockAddrIn& sin);
  60. /**
  61. * Clear struct
  62. */
  63. void Clear();
  64. /**
  65. * Compare NetworkAddr object
  66. * @param sin reference object
  67. * @return true if equal, otherwise false
  68. */
  69. bool IsEqual(const SockAddrIn& sin) const;
  70. /**
  71. * Check if NULL
  72. * @return true if object is 'NULL' - not valid
  73. */
  74. bool IsNull() const { return IsEqual(NULLAddr); }
  75. /**
  76. * Get Socket address family
  77. * @return Socket address family (IPv4: AF_INET, IPv6: AF_INET6)
  78. */
  79. short GetFamily () const { return ss_family; }
  80. /**
  81. * Get IP address (IPv4)
  82. * @return IPv4 network address in network format
  83. */
  84. ULONG GetIPAddr() const { return ((SOCKADDR_IN*)this)->sin_addr.s_addr; }
  85. /**
  86. * Get Port
  87. * @return Port number in network format
  88. */
  89. short GetPort() const { return ((SOCKADDR_IN*)this)->sin_port; }
  90. /**
  91. * Create from string. Service is service name or port number.
  92. * @param pszAddr Hostname or IP address
  93. * @param pszService Service name or port
  94. * @param nFamily Socket address family (default: AF_INET, AF_INET6)
  95. * @return true if object is now valid
  96. */
  97. bool CreateFrom(LPCTSTR pszAddr, LPCTSTR pszService, int nFamily = AF_INET);
  98. /**
  99. * Create from number. Initialize object from host or network format values (IPv4 only).
  100. * @param lIPAddr IP Address
  101. * @param nPort Port number
  102. * @param nFamily Socket address family (default: AF_INET)
  103. * @param bFmtHost flag to indicate that IP address and port are in host (true) or network format (false)
  104. * @return true if object is now valid
  105. */
  106. bool CreateFrom(ULONG lIPAddr, USHORT nPort, int nFamily = AF_INET, bool bFmtHost = true);
  107. /**
  108. * Object copy operator
  109. * @param sin reference object to copy from
  110. * @return reference of 'this' object
  111. */
  112. SockAddrIn& operator=(const SockAddrIn& sin) { return Copy( sin ); }
  113. /**
  114. * Equality operator
  115. * @param sin reference object to test
  116. * @return true if object is equal
  117. */
  118. bool operator==(const SockAddrIn& sin) const { return IsEqual( sin ); }
  119. /**
  120. * Not Equal operator
  121. * @param sin reference object to test
  122. * @return true if object is different
  123. */
  124. bool operator!=(const SockAddrIn& sin) const { return !IsEqual( sin ); }
  125. /**
  126. * SOCKADDR conversion. Return SOCKADDR of this object.
  127. */
  128. operator LPSOCKADDR() { return reinterpret_cast<LPSOCKADDR>(this); }
  129. /**
  130. * PIN6_ADDR conversion. Return const PIN6_ADDR of this object.
  131. */
  132. operator const IN6_ADDR*() const { return reinterpret_cast<const IN6_ADDR*>(this); }
  133. /**
  134. * PIN6_ADDR conversion. Return PIN6_ADDR of this object.
  135. */
  136. operator PIN6_ADDR() { return reinterpret_cast<PIN6_ADDR>(this); }
  137. /**
  138. * Size of this struct (SOCKADDR_IN) or (SOCKADDR_STORAGE) depending on address family
  139. * @return size of this structure (SOCKADDR_IN) or (SOCKADDR_STORAGE)
  140. */
  141. size_t Size() const { return (ss_family == AF_INET) ? sizeof(sockaddr_in) : sizeof(sockaddr_storage); }
  142. /**
  143. * Initialize this object from SOCKADDR_IN struct.
  144. * @param psin SOCKADDR_IN object pointer
  145. */
  146. void SetAddr(const sockaddr_in* psin) { SetAddr(reinterpret_cast<const sockaddr_storage*>(psin)); }
  147. /**
  148. * Initialize this object from SOCKADDR_IN6 struct.
  149. * @param psin SOCKADDR_IN6 object pointer
  150. */
  151. void SetAddr(const sockaddr_in6* psin) { SetAddr(reinterpret_cast<const sockaddr_storage*>(psin)); }
  152. /**
  153. * Initialize this object from SOCKADDR_STORAGE struct.
  154. * @param pss SOCKADDR_STORAGE object pointer
  155. */
  156. void SetAddr(const sockaddr_storage* pss) { ss_family = pss->ss_family; memcpy(this, pss, Size()); }
  157. static SockAddrIn NULLAddr; ///< Null Address
  158. };
  159. #ifdef WIN32
  160. typedef LPWSAOVERLAPPED_COMPLETION_ROUTINE LPWSACOMPLETIONROUTINE;
  161. #endif
  162. // Jeff.add.2014.08.07--------------------------------------------------------
  163. #define SOCKET_SERVER_PORT 6432
  164. // Jeff.注意, 这里单个包的大小不能超过该设置数值,一般设置为10*1024即10KB; 不过实际中可根据实际情况处理,如使用TCP来传输数据的话;
  165. #define SOCKET_BUFFSIZE 1024*10
  166. // Jeff.add.2014.08.07--------------------------------------------------------
  167. /**
  168. * CSocketHandle class
  169. * Socket communication class
  170. */
  171. class CSocketHandle
  172. {
  173. public:
  174. /**
  175. * Default constructor
  176. */
  177. CSocketHandle();
  178. /**
  179. * Destructor
  180. */
  181. ~CSocketHandle();
  182. /**
  183. * Check if socket is opened
  184. * @return true if this is opened (valid)
  185. */
  186. bool IsOpen() const;
  187. /**
  188. * Get SOCKET handle
  189. * @return SOCKET handle (handle is INVALID_SOCKET if object is closed)
  190. */
  191. SOCKET GetSocket() const;
  192. /**
  193. * Get SOCKET type
  194. * @return SOCKET type (-1 if not a valid socket)
  195. */
  196. int GetSocketType() const;
  197. /**
  198. * Attach a socket handle
  199. * This function may fail is socket handle is not valid or object already in used. Call Detach or Close to release object.
  200. * @param sock Socket handle to attach to this class
  201. * @return true if successful, otherwise false
  202. */
  203. bool Attach(SOCKET sock);
  204. /**
  205. * Detach a socket handle
  206. * @return previous socket handle or INVALID_SOCKET
  207. */
  208. SOCKET Detach();
  209. /**
  210. * Create a Socket - Server side.
  211. * @param pszHost hostname or IP address of adapter
  212. * @param pszServiceName Network service name or port number
  213. * @param nFamily address family to use (AF_INET, AF_INET6)
  214. * @param nType type of socket to create (SOCK_STREAM, SOCK_DGRAM)
  215. * @param uOptions Additional options (SO_BROADCAST, SO_REUSEADDR)
  216. * @return true if successful, otherwise false (call WSAGetLastError() to retrieve latest error)
  217. * @sa InitLibrary, ConnectTo, IsOpen
  218. */
  219. bool CreateSocket(LPCTSTR pszHost, LPCTSTR pszServiceName, int nFamily, int nType, UINT uOptions = 0);
  220. /**
  221. * Create a socket, connect to a server - Client side.
  222. * @param pszHostName Hostname or NIC address
  223. * @param pszRemote Remote network address
  224. * @param pszServiceName Network service name or port number
  225. * @param nFamily address family to use (AF_INET, AF_INET6)
  226. * @param nType type of socket to create (SOCK_STREAM, SOCK_DGRAM)
  227. * @return true if successful, otherwise false (call WSAGetLastError() to retrieve latest error)
  228. * @sa InitLibrary, CreateSocket, IsOpen
  229. */
  230. bool ConnectTo(LPCTSTR pszHostName, LPCTSTR pszRemote, LPCTSTR pszServiceName, int nFamily, int nType);
  231. /**
  232. * Close Socket
  233. * @sa InitLibrary, CreateSocket, ConnectTo, IsOpen
  234. */
  235. void Close();
  236. /**
  237. * Read from socket
  238. * @param lpBuffer Buffer to receive data
  239. * @param dwSize Size of buffer in bytes
  240. * @param lpAddrIn Peer address for UDP - this must be NULL for TCP
  241. * @param dwTimeout Read timeout in milliseconds
  242. * @return number of bytes read or (-1L) if fail
  243. * @sa InitLibrary, CreateSocket, ConnectTo, IsOpen, ReadEx, Write, WriteEx
  244. */
  245. DWORD Read(LPBYTE lpBuffer, DWORD dwSize, LPSOCKADDR lpAddrIn = NULL, DWORD dwTimeout = INFINITE);
  246. #ifdef WIN32
  247. /**
  248. * Read from socket (asynchronous mode).
  249. * @param lpBuffer Buffer to receive data
  250. * @param dwSize Size of buffer in bytes
  251. * @param lpAddrIn SockAddrIn for UDP - this must be NULL for TCP
  252. * @param lpOverlapped Windows Overlapped structure (required)
  253. * @param lpCompletionRoutine Winsock Completion routine (required)
  254. * @return number of bytes read, overlapped operation is pending or (-1L) if fail
  255. * @sa InitLibrary, CreateSocket, ConnectTo, IsOpen, Read, Write, WriteEx, IOControl, GetTransferOverlappedResult
  256. */
  257. DWORD ReadEx(LPBYTE lpBuffer, DWORD dwSize, LPSOCKADDR lpAddrIn,
  258. LPWSAOVERLAPPED lpOverlapped, LPWSACOMPLETIONROUTINE lpCompletionRoutine);
  259. #endif
  260. /**
  261. * Write to a destination socket
  262. * @param lpBuffer Buffer to send
  263. * @param dwCount Number of bytes to send
  264. * @param lpAddrIn Peer address for UDP - this must be NULL for TCP
  265. * @param dwTimeout Write timeout in milliseconds
  266. * @return number of bytes sent or (-1L) if fail
  267. * @sa InitLibrary, CreateSocket, ConnectTo, IsOpen, Read, ReadEx, WriteEx
  268. */
  269. DWORD Write(const LPBYTE lpBuffer, DWORD dwCount, const LPSOCKADDR lpAddrIn = NULL, DWORD dwTimeout = INFINITE);
  270. #ifdef WIN32
  271. /**
  272. * Write to a destination socket (asynchronous mode).
  273. * @param lpBuffer Buffer to send
  274. * @param dwCount Number of bytes to send
  275. * @param lpAddrIn SockAddrIn for UDP - this must be NULL for TCP
  276. * @param lpOverlapped Windows Overlapped structure (required)
  277. * @param lpCompletionRoutine Winsock Completion routine (required)
  278. * @return number of bytes read, overlapped operation is pending or (-1L) if fail
  279. * @sa InitLibrary, CreateSocket, ConnectTo, IsOpen, Read, ReadEx, Write, IOControl, GetTransferOverlappedResult
  280. */
  281. DWORD WriteEx(const LPBYTE lpBuffer, DWORD dwCount, const LPSOCKADDR lpAddrIn,
  282. LPWSAOVERLAPPED lpOverlapped, LPWSACOMPLETIONROUTINE lpCompletionRoutine);
  283. #endif
  284. #ifdef WIN32
  285. /**
  286. * Control the mode of a socket (asynchronous mode)
  287. * @param dwIoCode Control code of operation to perform
  288. * @param lpInBuffer Pointer to the input buffer
  289. * @param cbInBuffer Size of the input buffer, in bytes
  290. * @param lpOutBuffer Pointer to the output buffer
  291. * @param cbOutBuffer Size of the output buffer, in bytes
  292. * @param lpcbBytesReturned Pointer to actual number of bytes of output
  293. * @param lpOverlapped Winsock Overlapped structure
  294. * @param lpCompletionRoutine Winsock Completion routine
  295. * @return true if successful, otherwise false (call WSAGetLastError() to retrieve latest error)
  296. * @sa InitLibrary, CreateSocket, ConnectTo, IsOpen, ReadEx, WriteEx, GetTransferOverlappedResult
  297. */
  298. bool IOControl(DWORD dwIoCode, LPBYTE lpInBuffer, DWORD cbInBuffer,
  299. LPBYTE lpOutBuffer, DWORD cbOutBuffer,
  300. LPDWORD lpcbBytesReturned, LPWSAOVERLAPPED lpOverlapped,
  301. LPWSACOMPLETIONROUTINE lpCompletionRoutine);
  302. /**
  303. * Get Overlapped result (asynchronous mode)
  304. * @param lpOverlapped Windows Overlapped structure (required)
  305. * @param lpcbTransfer Pointer to get number of bytes transferred
  306. * @param bWait Force wait for overlapped operation to complete
  307. * @param lpdwFlags Optional flags (see MSDN on WSARecv API)
  308. * @sa InitLibrary, CreateSocket, ConnectTo, IsOpen, ReadEx, WriteEx, IOControl
  309. */
  310. bool GetTransferOverlappedResult(LPWSAOVERLAPPED lpOverlapped, LPDWORD lpcbTransfer, bool bWait = true, LPDWORD lpdwFlags = 0);
  311. #endif
  312. /**
  313. * Get Socket name - get local address
  314. * @param saddr_in current local address and port when connected
  315. * @return true if successful
  316. * @sa InitLibrary
  317. */
  318. bool GetSockName(SockAddrIn& saddr_in) const;
  319. /**
  320. * Get Peer Socket name - get peer address
  321. * @param saddr_in peer address and port (use only with TCP or client mode UDP)
  322. * @return true if successful
  323. * @sa InitLibrary
  324. */
  325. bool GetPeerName(SockAddrIn& saddr_in) const;
  326. /**
  327. * Register to multicast address
  328. * @param pszIPAddr multicast IP group
  329. * @param pszNIC interface IP address
  330. * @return true if successful
  331. * @sa InitLibrary
  332. */
  333. bool AddMembership(LPCTSTR pszIPAddr, LPCTSTR pszNIC);
  334. /**
  335. * Unregister from a multicast address
  336. * @param pszIPAddr multicast IP group
  337. * @param pszNIC interface IP address
  338. * @return true if successful
  339. * @sa InitLibrary
  340. */
  341. bool DropMembership(LPCTSTR pszIPAddr, LPCTSTR pszNIC);
  342. /**
  343. * Initialize Winsock library. This function calls WSAStartup.
  344. * @param wVersion Winsock version use MAKEWORD macro if possible (e.g.: MAKEWORD(2,2))
  345. * @return true if successful
  346. * @sa ReleaseLibrary
  347. */
  348. static bool InitLibrary(WORD wVersion);
  349. /**
  350. * Release Winsock library
  351. * @return true if successful
  352. * @sa InitLibrary
  353. */
  354. static bool ReleaseLibrary();
  355. /**
  356. * Wait for a new connection
  357. * @param sock A TCP socket handle. A new socket is return returned.
  358. * @return A new socket when a new client connects
  359. * @sa GetSocket, CreateSocket
  360. */
  361. static SOCKET WaitForConnection(SOCKET sock);
  362. /**
  363. * Shutdown a connection
  364. * @param sock Socket to shutdown communication
  365. * @return true if successful
  366. */
  367. static bool ShutdownConnection(SOCKET sock);
  368. /**
  369. * Check if IP address is unicast (network order).
  370. * @param ulAddr IP address (expected valid unicast address)
  371. * @return true if successful
  372. */
  373. static bool IsUnicastIP( ULONG ulAddr );
  374. /**
  375. * Check if IP address is multicast (network order).
  376. * @param ulAddr IP address (expected valid multicast address)
  377. * @return true if successful
  378. */
  379. static bool IsMulticastIP( ULONG ulAddr );
  380. /**
  381. * Format IP address to string
  382. * @param pszIPAddr Buffer to hold string
  383. * @param nSize Size of buffer in characters
  384. * @param ulAddr IP Address to format
  385. * @param bFmtHost Specify if address (ulAddr) is in host (true) or network format (false)
  386. * @return true if successful. Possible error could be INSUFFICIENT_BUFFER
  387. */
  388. static bool FormatIP(LPTSTR pszIPAddr, UINT nSize, ULONG ulAddr, bool bFmtHost);
  389. /**
  390. * Format IP address to string
  391. * @param pszIPAddr Buffer to hold string
  392. * @param nSize Size of buffer in characters
  393. * @param addrIn IP Address to format
  394. * @return true if successful. Possible error could be INSUFFICIENT_BUFFER
  395. */
  396. static bool FormatIP(LPTSTR pszIPAddr, UINT nSize, const SockAddrIn& addrIn);
  397. /**
  398. * Get service port number
  399. * @param pszServiceName Network service name (e.g.: "ftp", "telnet") or port number
  400. * @return port number
  401. */
  402. static USHORT GetPortNumber( LPCTSTR pszServiceName );
  403. /**
  404. * Get IP address of a host
  405. * @param pszHostName host name or IP address
  406. * @return Host IP address in host format
  407. * @sa GetAddressInfo
  408. */
  409. static ULONG GetIPAddress( LPCTSTR pszHostName );
  410. /**
  411. * Get current localname for this machine
  412. * @param pszName Buffer to receive host name
  413. * @param nSize Size of this buffer in character
  414. * @return true if successful
  415. */
  416. static bool GetLocalName(LPTSTR pszName, UINT nSize);
  417. /**
  418. * Get current (default) IP address for this machine
  419. * @param pszAddress Buffer to receive IP address (IPv4, IPv6 format)
  420. * @param nSize Size of this buffer in character
  421. * @param nFamily address family to use (AF_INET, AF_INET6)
  422. * @return true if successful
  423. */
  424. static bool GetLocalAddress(LPTSTR pszAddress, UINT nSize, int nFamily = AF_INET);
  425. /**
  426. * Get IP address info of a host (Supports: IPv4 and IPv6)
  427. * @param pszHostName host name or IP address
  428. * @param pszServiceName pszServiceName Network service name (e.g.: "ftp", "telnet") or port number
  429. * @param nFamily address family to use (AF_INET, AF_INET6)
  430. * @param sockAddr Socket address to fill in
  431. * @return true if successful
  432. */
  433. static bool GetAddressInfo( LPCTSTR pszHostName, LPCTSTR pszServiceName, int nFamily, SockAddrIn& sockAddr);
  434. // CSocketHandle - data
  435. protected:
  436. SOCKET m_hSocket; ///< socket handle
  437. // Jeff.add.2014.08.07-----------------------------------.
  438. public:
  439. unsigned char m_szpendingbuf[SOCKET_BUFFSIZE];
  440. unsigned int m_npendingSize;
  441. CRITICAL_SECTION m_hClient2SrvSection;
  442. // Jeff.add.2014.08.07-----------------------------------.
  443. };
  444. /** @}*/
  445. #endif //SOCKETHANDLE_H