TcpClient.h 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222
  1. /*
  2. * Copyright: JessMA Open Source (ldcsaa@gmail.com)
  3. *
  4. * Version : 4.2.1
  5. * Author : Bruce Liang
  6. * Website : http://www.jessma.org
  7. * Project : https://github.com/ldcsaa
  8. * Blog : http://www.cnblogs.com/ldcsaa
  9. * Wiki : http://www.oschina.net/p/hp-socket
  10. * QQ Group : 75375912
  11. *
  12. * Licensed under the Apache License, Version 2.0 (the "License");
  13. * you may not use this file except in compliance with the License.
  14. * You may obtain a copy of the License at
  15. *
  16. * http://www.apache.org/licenses/LICENSE-2.0
  17. *
  18. * Unless required by applicable law or agreed to in writing, software
  19. * distributed under the License is distributed on an "AS IS" BASIS,
  20. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  21. * See the License for the specific language governing permissions and
  22. * limitations under the License.
  23. */
  24. #pragma once
  25. #include "SocketHelper.h"
  26. #include "../../Common/Src/Event.h"
  27. #include "../../Common/Src/bufferptr.h"
  28. #include "../../Common/Src/bufferpool.h"
  29. #include "../../Common/Src/CriticalSection.h"
  30. class CTcpClient : public ITcpClient
  31. {
  32. public:
  33. virtual BOOL Start (LPCTSTR lpszRemoteAddress, USHORT usPort, BOOL bAsyncConnect = TRUE, LPCTSTR lpszBindAddress = nullptr);
  34. virtual BOOL Stop ();
  35. virtual BOOL Send (const BYTE* pBuffer, int iLength, int iOffset = 0);
  36. virtual BOOL SendSmallFile (LPCTSTR lpszFileName, const LPWSABUF pHead = nullptr, const LPWSABUF pTail = nullptr);
  37. virtual BOOL SendPackets (const WSABUF pBuffers[], int iCount) {return DoSendPackets(pBuffers, iCount);}
  38. virtual BOOL HasStarted () {return m_enState == SS_STARTED || m_enState == SS_STARTING;}
  39. virtual EnServiceState GetState () {return m_enState;}
  40. virtual CONNID GetConnectionID () {return m_dwConnID;};
  41. virtual EnSocketError GetLastError () {return m_enLastError;}
  42. virtual LPCTSTR GetLastErrorDesc () {return ::GetSocketErrorDesc(m_enLastError);}
  43. virtual BOOL GetLocalAddress (TCHAR lpszAddress[], int& iAddressLen, USHORT& usPort);
  44. virtual BOOL GetRemoteHost (TCHAR lpszHost[], int& iHostLen, USHORT& usPort);
  45. virtual BOOL GetPendingDataLength (int& iPending) {iPending = m_iPending; return HasStarted();}
  46. public:
  47. virtual BOOL IsSecure () {return FALSE;}
  48. virtual void SetSocketBufferSize (DWORD dwSocketBufferSize) {m_dwSocketBufferSize = dwSocketBufferSize;}
  49. virtual void SetKeepAliveTime (DWORD dwKeepAliveTime) {m_dwKeepAliveTime = dwKeepAliveTime;}
  50. virtual void SetKeepAliveInterval (DWORD dwKeepAliveInterval) {m_dwKeepAliveInterval = dwKeepAliveInterval;}
  51. virtual void SetFreeBufferPoolSize (DWORD dwFreeBufferPoolSize) {m_dwFreeBufferPoolSize = dwFreeBufferPoolSize;}
  52. virtual void SetFreeBufferPoolHold (DWORD dwFreeBufferPoolHold) {m_dwFreeBufferPoolHold = dwFreeBufferPoolHold;}
  53. virtual void SetExtra (PVOID pExtra) {m_pExtra = pExtra;}
  54. virtual DWORD GetSocketBufferSize () {return m_dwSocketBufferSize;}
  55. virtual DWORD GetKeepAliveTime () {return m_dwKeepAliveTime;}
  56. virtual DWORD GetKeepAliveInterval () {return m_dwKeepAliveInterval;}
  57. virtual DWORD GetFreeBufferPoolSize () {return m_dwFreeBufferPoolSize;}
  58. virtual DWORD GetFreeBufferPoolHold () {return m_dwFreeBufferPoolHold;}
  59. virtual PVOID GetExtra () {return m_pExtra;}
  60. protected:
  61. virtual EnHandleResult FirePrepareConnect(SOCKET socket)
  62. {return DoFirePrepareConnect(this, socket);}
  63. virtual EnHandleResult FireConnect()
  64. {
  65. EnHandleResult rs = DoFireConnect(this);
  66. if(rs != HR_ERROR) rs = FireHandShake();
  67. return rs;
  68. }
  69. virtual EnHandleResult FireHandShake()
  70. {return DoFireHandShake(this);}
  71. virtual EnHandleResult FireSend(const BYTE* pData, int iLength)
  72. {return DoFireSend(this, pData, iLength);}
  73. virtual EnHandleResult FireReceive(const BYTE* pData, int iLength)
  74. {return DoFireReceive(this, pData, iLength);}
  75. virtual EnHandleResult FireReceive(int iLength)
  76. {return DoFireReceive(this, iLength);}
  77. virtual EnHandleResult FireClose(EnSocketOperation enOperation, int iErrorCode)
  78. {return DoFireClose(this, enOperation, iErrorCode);}
  79. virtual EnHandleResult DoFirePrepareConnect(ITcpClient* pSender, SOCKET socket)
  80. {return m_pListener->OnPrepareConnect(pSender, pSender->GetConnectionID(), socket);}
  81. virtual EnHandleResult DoFireConnect(ITcpClient* pSender)
  82. {return m_pListener->OnConnect(pSender, pSender->GetConnectionID());}
  83. virtual EnHandleResult DoFireHandShake(ITcpClient* pSender)
  84. {return m_pListener->OnHandShake(pSender, pSender->GetConnectionID());}
  85. virtual EnHandleResult DoFireSend(ITcpClient* pSender, const BYTE* pData, int iLength)
  86. {return m_pListener->OnSend(pSender, pSender->GetConnectionID(), pData, iLength);}
  87. virtual EnHandleResult DoFireReceive(ITcpClient* pSender, const BYTE* pData, int iLength)
  88. {return m_pListener->OnReceive(pSender, pSender->GetConnectionID(), pData, iLength);}
  89. virtual EnHandleResult DoFireReceive(ITcpClient* pSender, int iLength)
  90. {return m_pListener->OnReceive(pSender, pSender->GetConnectionID(), iLength);}
  91. virtual EnHandleResult DoFireClose(ITcpClient* pSender, EnSocketOperation enOperation, int iErrorCode)
  92. {return m_pListener->OnClose(pSender, pSender->GetConnectionID(), enOperation, iErrorCode);}
  93. void SetLastError(EnSocketError code, LPCSTR func, int ec);
  94. virtual BOOL CheckParams();
  95. virtual void PrepareStart();
  96. virtual void Reset();
  97. virtual void OnWorkerThreadEnd(DWORD dwThreadID) {}
  98. BOOL DoSendPackets(const WSABUF pBuffers[], int iCount);
  99. static BOOL DoSendPackets(CTcpClient* pClient, const WSABUF pBuffers[], int iCount)
  100. {return pClient->DoSendPackets(pBuffers, iCount);}
  101. protected:
  102. void SetReserved (PVOID pReserved) {m_pReserved = pReserved;}
  103. PVOID GetReserved () {return m_pReserved;}
  104. BOOL GetRemoteHost (LPCSTR* lpszHost, USHORT* pusPort = nullptr);
  105. private:
  106. void SetRemoteHost (LPCTSTR lpszHost, USHORT usPort);
  107. BOOL CheckStarting();
  108. BOOL CheckStoping(DWORD dwCurrentThreadID);
  109. BOOL CreateClientSocket();
  110. BOOL BindClientSocket(LPCTSTR lpszBindAddress);
  111. BOOL ConnectToServer(LPCTSTR lpszRemoteAddress, USHORT usPort);
  112. BOOL CreateWorkerThread();
  113. BOOL ProcessNetworkEvent();
  114. BOOL ReadData();
  115. BOOL SendData();
  116. BOOL DoSendData(TItem* pItem);
  117. TItem* GetSendBuffer();
  118. BOOL SendInternal(const WSABUF pBuffers[], int iCount);
  119. void WaitForWorkerThreadEnd(DWORD dwCurrentThreadID);
  120. BOOL HandleError (WSANETWORKEVENTS& events);
  121. BOOL HandleRead (WSANETWORKEVENTS& events);
  122. BOOL HandleWrite (WSANETWORKEVENTS& events);
  123. BOOL HandleConnect (WSANETWORKEVENTS& events);
  124. BOOL HandleClose (WSANETWORKEVENTS& events);
  125. static UINT WINAPI WorkerThreadProc(LPVOID pv);
  126. public:
  127. CTcpClient(ITcpClientListener* pListener)
  128. : m_pListener (pListener)
  129. , m_lsSend (m_itPool)
  130. , m_soClient (INVALID_SOCKET)
  131. , m_evSocket (nullptr)
  132. , m_dwConnID (0)
  133. , m_usPort (0)
  134. , m_hWorker (nullptr)
  135. , m_dwWorkerID (0)
  136. , m_bAsyncConnect (FALSE)
  137. , m_iPending (0)
  138. , m_enState (SS_STOPPED)
  139. , m_enLastError (SE_OK)
  140. , m_pExtra (nullptr)
  141. , m_pReserved (nullptr)
  142. , m_dwSocketBufferSize (DEFAULT_TCP_SOCKET_BUFFER_SIZE)
  143. , m_dwFreeBufferPoolSize(DEFAULT_CLIENT_FREE_BUFFER_POOL_SIZE)
  144. , m_dwFreeBufferPoolHold(DEFAULT_CLIENT_FREE_BUFFER_POOL_HOLD)
  145. , m_dwKeepAliveTime (DEFALUT_TCP_KEEPALIVE_TIME)
  146. , m_dwKeepAliveInterval (DEFALUT_TCP_KEEPALIVE_INTERVAL)
  147. {
  148. ASSERT(m_wsSocket.IsValid());
  149. ASSERT(m_pListener);
  150. }
  151. virtual ~CTcpClient()
  152. {
  153. Stop();
  154. }
  155. private:
  156. CInitSocket m_wsSocket;
  157. private:
  158. ITcpClientListener* m_pListener;
  159. TClientCloseContext m_ccContext;
  160. BOOL m_bAsyncConnect;
  161. SOCKET m_soClient;
  162. HANDLE m_evSocket;
  163. CONNID m_dwConnID;
  164. DWORD m_dwSocketBufferSize;
  165. DWORD m_dwFreeBufferPoolSize;
  166. DWORD m_dwFreeBufferPoolHold;
  167. DWORD m_dwKeepAliveTime;
  168. DWORD m_dwKeepAliveInterval;
  169. HANDLE m_hWorker;
  170. UINT m_dwWorkerID;
  171. volatile EnServiceState m_enState;
  172. EnSocketError m_enLastError;
  173. PVOID m_pExtra;
  174. PVOID m_pReserved;
  175. CBufferPtr m_rcBuffer;
  176. protected:
  177. CStringA m_strHost;
  178. USHORT m_usPort;
  179. CItemPool m_itPool;
  180. private:
  181. CSpinGuard m_csState;
  182. CCriSec m_csSend;
  183. TItemList m_lsSend;
  184. CEvt m_evBuffer;
  185. CEvt m_evWorker;
  186. volatile int m_iPending;
  187. };