123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296 |
- /*
- * Copyright: JessMA Open Source (ldcsaa@gmail.com)
- *
- * Version : 3.6.1
- * Author : Bruce Liang
- * Website : http://www.jessma.org
- * Project : https://github.com/ldcsaa
- * Blog : http://www.cnblogs.com/ldcsaa
- * Wiki : http://www.oschina.net/p/hp-socket
- * QQ Group : 75375912
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
- #pragma once
- #include "SocketHelper.h"
- #include "../../Common/Src/Event.h"
- #include "../../Common/Src/STLHelper.h"
- #include "../../Common/Src/RingBuffer.h"
- #include "../../Common/Src/PrivateHeap.h"
- class CTcpServer : public ITcpServer
- {
- public:
- CTcpServer(ITcpServerListener* pListener)
- : m_pListener (pListener)
- , m_hCompletePort (nullptr)
- , m_soListen (INVALID_SOCKET)
- , m_iRemainAcceptSockets (0)
- , m_pfnAcceptEx (nullptr)
- , m_pfnGetAcceptExSockaddrs (nullptr)
- , m_pfnDisconnectEx (nullptr)
- , m_enLastError (SE_OK)
- , m_enState (SS_STOPPED)
- , m_enSendPolicy (SP_PACK)
- , m_dwMaxConnectionCount (DEFAULT_MAX_CONNECTION_COUNT)
- , m_dwWorkerThreadCount (DEFAULT_WORKER_THREAD_COUNT)
- , m_dwSocketListenQueue (DEFAULT_TCP_SERVER_SOCKET_LISTEN_QUEUE)
- , m_dwAcceptSocketCount (DEFAULT_TCP_SERVER_ACCEPT_SOCKET_COUNT)
- , m_dwSocketBufferSize (DEFAULT_TCP_SOCKET_BUFFER_SIZE)
- , m_dwFreeSocketObjLockTime (DEFAULT_FREE_SOCKETOBJ_LOCK_TIME)
- , m_dwFreeSocketObjPool (DEFAULT_FREE_SOCKETOBJ_POOL)
- , m_dwFreeBufferObjPool (DEFAULT_FREE_BUFFEROBJ_POOL)
- , m_dwFreeSocketObjHold (DEFAULT_FREE_SOCKETOBJ_HOLD)
- , m_dwFreeBufferObjHold (DEFAULT_FREE_BUFFEROBJ_HOLD)
- , m_dwKeepAliveTime (DEFALUT_TCP_KEEPALIVE_TIME)
- , m_dwKeepAliveInterval (DEFALUT_TCP_KEEPALIVE_INTERVAL)
- , m_bMarkSilence (TRUE)
- {
- ASSERT(m_wsSocket.IsValid());
- ASSERT(m_pListener);
- }
- virtual ~CTcpServer()
- {
- Stop();
- }
- public:
- virtual BOOL Start (LPCTSTR lpszBindAddress, USHORT usPort);
- virtual BOOL Stop ();
- virtual BOOL Send (CONNID dwConnID, const BYTE* pBuffer, int iLength, int iOffset = 0);
- virtual BOOL SendSmallFile (CONNID dwConnID, LPCTSTR lpszFileName, const LPWSABUF pHead = nullptr, const LPWSABUF pTail = nullptr);
- virtual BOOL SendPackets (CONNID dwConnID, const WSABUF pBuffers[], int iCount) {return DoSendPackets(dwConnID, pBuffers, iCount);}
- virtual BOOL HasStarted () {return m_enState == SS_STARTED || m_enState == SS_STARTING;}
- virtual EnServiceState GetState () {return m_enState;}
- virtual BOOL Disconnect (CONNID dwConnID, BOOL bForce = TRUE);
- virtual BOOL DisconnectLongConnections (DWORD dwPeriod, BOOL bForce = TRUE);
- virtual BOOL DisconnectSilenceConnections(DWORD dwPeriod, BOOL bForce = TRUE);
- virtual BOOL GetListenAddress (TCHAR lpszAddress[], int& iAddressLen, USHORT& usPort);
- virtual BOOL GetLocalAddress (CONNID dwConnID, TCHAR lpszAddress[], int& iAddressLen, USHORT& usPort);
- virtual BOOL GetRemoteAddress (CONNID dwConnID, TCHAR lpszAddress[], int& iAddressLen, USHORT& usPort);
-
- virtual BOOL GetPendingDataLength (CONNID dwConnID, int& iPending);
- virtual DWORD GetConnectionCount ();
- virtual BOOL GetAllConnectionIDs (CONNID pIDs[], DWORD& dwCount);
- virtual BOOL GetConnectPeriod (CONNID dwConnID, DWORD& dwPeriod);
- virtual BOOL GetSilencePeriod (CONNID dwConnID, DWORD& dwPeriod);
- virtual EnSocketError GetLastError () {return m_enLastError;}
- virtual LPCTSTR GetLastErrorDesc () {return ::GetSocketErrorDesc(m_enLastError);}
- public:
- virtual BOOL SetConnectionExtra(CONNID dwConnID, PVOID pExtra);
- virtual BOOL GetConnectionExtra(CONNID dwConnID, PVOID* ppExtra);
- virtual void SetSendPolicy (EnSendPolicy enSendPolicy) {m_enSendPolicy = enSendPolicy;}
- virtual void SetMaxConnectionCount (DWORD dwMaxConnectionCount) {m_dwMaxConnectionCount = dwMaxConnectionCount;}
- virtual void SetWorkerThreadCount (DWORD dwWorkerThreadCount) {m_dwWorkerThreadCount = dwWorkerThreadCount;}
- virtual void SetSocketListenQueue (DWORD dwSocketListenQueue) {m_dwSocketListenQueue = dwSocketListenQueue;}
- virtual void SetAcceptSocketCount (DWORD dwAcceptSocketCount) {m_dwAcceptSocketCount = dwAcceptSocketCount;}
- virtual void SetSocketBufferSize (DWORD dwSocketBufferSize) {m_dwSocketBufferSize = dwSocketBufferSize;}
- virtual void SetFreeSocketObjLockTime (DWORD dwFreeSocketObjLockTime) {m_dwFreeSocketObjLockTime = dwFreeSocketObjLockTime;}
- virtual void SetFreeSocketObjPool (DWORD dwFreeSocketObjPool) {m_dwFreeSocketObjPool = dwFreeSocketObjPool;}
- virtual void SetFreeBufferObjPool (DWORD dwFreeBufferObjPool) {m_dwFreeBufferObjPool = dwFreeBufferObjPool;}
- virtual void SetFreeSocketObjHold (DWORD dwFreeSocketObjHold) {m_dwFreeSocketObjHold = dwFreeSocketObjHold;}
- virtual void SetFreeBufferObjHold (DWORD dwFreeBufferObjHold) {m_dwFreeBufferObjHold = dwFreeBufferObjHold;}
- virtual void SetKeepAliveTime (DWORD dwKeepAliveTime) {m_dwKeepAliveTime = dwKeepAliveTime;}
- virtual void SetKeepAliveInterval (DWORD dwKeepAliveInterval) {m_dwKeepAliveInterval = dwKeepAliveInterval;}
- virtual void SetMarkSilence (BOOL bMarkSilence) {m_bMarkSilence = bMarkSilence;}
- virtual EnSendPolicy GetSendPolicy () {return m_enSendPolicy;}
- virtual DWORD GetMaxConnectionCount () {return m_dwMaxConnectionCount;}
- virtual DWORD GetWorkerThreadCount () {return m_dwWorkerThreadCount;}
- virtual DWORD GetSocketListenQueue () {return m_dwSocketListenQueue;}
- virtual DWORD GetAcceptSocketCount () {return m_dwAcceptSocketCount;}
- virtual DWORD GetSocketBufferSize () {return m_dwSocketBufferSize;}
- virtual DWORD GetFreeSocketObjLockTime () {return m_dwFreeSocketObjLockTime;}
- virtual DWORD GetFreeSocketObjPool () {return m_dwFreeSocketObjPool;}
- virtual DWORD GetFreeBufferObjPool () {return m_dwFreeBufferObjPool;}
- virtual DWORD GetFreeSocketObjHold () {return m_dwFreeSocketObjHold;}
- virtual DWORD GetFreeBufferObjHold () {return m_dwFreeBufferObjHold;}
- virtual DWORD GetKeepAliveTime () {return m_dwKeepAliveTime;}
- virtual DWORD GetKeepAliveInterval () {return m_dwKeepAliveInterval;}
- virtual BOOL IsMarkSilence () {return m_bMarkSilence;}
- protected:
- virtual EnHandleResult FirePrepareListen(SOCKET soListen)
- {return DoFirePrepareListen(soListen);}
- virtual EnHandleResult FireAccept(TSocketObj* pSocketObj)
- {
- EnHandleResult rs = DoFireAccept(pSocketObj);
- if(rs != HR_ERROR) rs = FireHandShake(pSocketObj);
- return rs;
- }
- virtual EnHandleResult FireHandShake(TSocketObj* pSocketObj)
- {return DoFireHandShake(pSocketObj);}
- virtual EnHandleResult FireReceive(TSocketObj* pSocketObj, const BYTE* pData, int iLength)
- {return DoFireReceive(pSocketObj, pData, iLength);}
- virtual EnHandleResult FireReceive(TSocketObj* pSocketObj, int iLength)
- {return DoFireReceive(pSocketObj, iLength);}
- virtual EnHandleResult FireSend(TSocketObj* pSocketObj, const BYTE* pData, int iLength)
- {return DoFireSend(pSocketObj, pData, iLength);}
- virtual EnHandleResult FireClose(TSocketObj* pSocketObj, EnSocketOperation enOperation, int iErrorCode)
- {return DoFireClose(pSocketObj, enOperation, iErrorCode);}
- virtual EnHandleResult FireShutdown()
- {return DoFireShutdown();}
- virtual EnHandleResult DoFirePrepareListen(SOCKET soListen)
- {return m_pListener->OnPrepareListen(soListen);}
- virtual EnHandleResult DoFireAccept(TSocketObj* pSocketObj)
- {return m_pListener->OnAccept(pSocketObj->connID, pSocketObj->socket);}
- virtual EnHandleResult DoFireHandShake(TSocketObj* pSocketObj)
- {return m_pListener->OnHandShake(pSocketObj->connID);}
- virtual EnHandleResult DoFireReceive(TSocketObj* pSocketObj, const BYTE* pData, int iLength)
- {return m_pListener->OnReceive(pSocketObj->connID, pData, iLength);}
- virtual EnHandleResult DoFireReceive(TSocketObj* pSocketObj, int iLength)
- {return m_pListener->OnReceive(pSocketObj->connID, iLength);}
- virtual EnHandleResult DoFireSend(TSocketObj* pSocketObj, const BYTE* pData, int iLength)
- {return m_pListener->OnSend(pSocketObj->connID, pData, iLength);}
- virtual EnHandleResult DoFireClose(TSocketObj* pSocketObj, EnSocketOperation enOperation, int iErrorCode)
- {return m_pListener->OnClose(pSocketObj->connID, enOperation, iErrorCode);}
- virtual EnHandleResult DoFireShutdown()
- {return m_pListener->OnShutdown();}
- void SetLastError(EnSocketError code, LPCSTR func, int ec);
- virtual BOOL CheckParams();
- virtual void PrepareStart();
- virtual void Reset();
- virtual void OnWorkerThreadEnd(DWORD dwThreadID) {}
- BOOL DoSendPackets(CONNID dwConnID, const WSABUF pBuffers[], int iCount);
- BOOL DoSendPackets(TSocketObj* pSocketObj, const WSABUF pBuffers[], int iCount);
- TSocketObj* FindSocketObj(CONNID dwConnID);
- private:
- EnHandleResult TriggerFireAccept(TSocketObj* pSocketObj);
- EnHandleResult TriggerFireReceive(TSocketObj* pSocketObj, TBufferObj* pBufferObj);
- EnHandleResult TriggerFireSend(TSocketObj* pSocketObj, TBufferObj* pBufferObj);
- EnHandleResult TriggerFireClose(TSocketObj* pSocketObj, EnSocketOperation enOperation, int iErrorCode);
- protected:
- BOOL SetConnectionExtra(TSocketObj* pSocketObj, PVOID pExtra);
- BOOL GetConnectionExtra(TSocketObj* pSocketObj, PVOID* ppExtra);
- BOOL SetConnectionReserved(CONNID dwConnID, PVOID pReserved);
- BOOL GetConnectionReserved(CONNID dwConnID, PVOID* ppReserved);
- BOOL SetConnectionReserved(TSocketObj* pSocketObj, PVOID pReserved);
- BOOL GetConnectionReserved(TSocketObj* pSocketObj, PVOID* ppReserved);
- BOOL SetConnectionReserved2(CONNID dwConnID, PVOID pReserved2);
- BOOL GetConnectionReserved2(CONNID dwConnID, PVOID* ppReserved2);
- BOOL SetConnectionReserved2(TSocketObj* pSocketObj, PVOID pReserved2);
- BOOL GetConnectionReserved2(TSocketObj* pSocketObj, PVOID* ppReserved2);
- private:
- BOOL CheckStarting();
- BOOL CheckStoping();
- BOOL CreateListenSocket(LPCTSTR lpszBindAddress, USHORT usPort);
- BOOL CreateCompletePort();
- BOOL CreateWorkerThreads();
- BOOL StartAccept();
- void CloseListenSocket();
- void WaitForAcceptSocketClose();
- void DisconnectClientSocket();
- void WaitForClientSocketClose();
- void ReleaseClientSocket();
- void ReleaseFreeSocket();
- void ReleaseFreeBuffer();
- void WaitForWorkerThreadEnd();
- void CloseCompletePort();
- TBufferObj* GetFreeBufferObj(int iLen = -1);
- TSocketObj* GetFreeSocketObj(CONNID dwConnID, SOCKET soClient);
- void AddFreeBufferObj(TBufferObj* pBufferObj);
- void AddFreeSocketObj(TSocketObj* pSocketObj, EnSocketCloseFlag enFlag = SCF_NONE, EnSocketOperation enOperation = SO_UNKNOWN, int iErrorCode = 0);
- TSocketObj* CreateSocketObj();
- void DeleteSocketObj(TSocketObj* pSocketObj);
- BOOL InvalidSocketObj(TSocketObj* pSocketObj);
- void ReleaseGCSocketObj(BOOL bForce = FALSE);
- void AddClientSocketObj(CONNID dwConnID, TSocketObj* pSocketObj);
- void CloseClientSocketObj(TSocketObj* pSocketObj, EnSocketCloseFlag enFlag = SCF_NONE, EnSocketOperation enOperation = SO_UNKNOWN, int iErrorCode = 0, int iShutdownFlag = SD_SEND);
- private:
- static UINT WINAPI WorkerThreadProc(LPVOID pv);
- EnIocpAction CheckIocpCommand(OVERLAPPED* pOverlapped, DWORD dwBytes, ULONG_PTR ulCompKey);
- void ForceDisconnect(CONNID dwConnID);
- void HandleIo (CONNID dwConnID, TSocketObj* pSocketObj, TBufferObj* pBufferObj, DWORD dwBytes, DWORD dwErrorCode);
- void HandleError (CONNID dwConnID, TSocketObj* pSocketObj, TBufferObj* pBufferObj, DWORD dwErrorCode);
- void HandleAccept (SOCKET soListen, TBufferObj* pBufferObj);
- void HandleSend (CONNID dwConnID, TSocketObj* pSocketObj, TBufferObj* pBufferObj);
- void HandleReceive (CONNID dwConnID, TSocketObj* pSocketObj, TBufferObj* pBufferObj);
- int SendInternal(TSocketObj* pSocketObj, const WSABUF pBuffers[], int iCount);
- int SendPack (TSocketObj* pSocketObj, const BYTE* pBuffer, int iLength);
- int SendSafe (TSocketObj* pSocketObj, const BYTE* pBuffer, int iLength);
- int SendDirect (TSocketObj* pSocketObj, const BYTE* pBuffer, int iLength);
- int CatAndPost (TSocketObj* pSocketObj, const BYTE* pBuffer, int iLength, BOOL isPostSend);
- BOOL DoAccept ();
- int DoReceive (CONNID dwConnID, TSocketObj* pSocketObj, TBufferObj* pBufferObj);
- int DoSend (CONNID dwConnID);
- int DoSend (TSocketObj* pSocketObj);
- int DoSendPack (TSocketObj* pSocketObj);
- int DoSendSafe (TSocketObj* pSocketObj);
- int SendItem (TSocketObj* pSocketObj);
- void CheckError (TSocketObj* pSocketObj, EnSocketOperation enOperation, int iErrorCode);
- private:
- EnSendPolicy m_enSendPolicy;
- DWORD m_dwMaxConnectionCount;
- DWORD m_dwWorkerThreadCount;
- DWORD m_dwSocketListenQueue;
- DWORD m_dwAcceptSocketCount;
- DWORD m_dwSocketBufferSize;
- DWORD m_dwFreeSocketObjLockTime;
- DWORD m_dwFreeSocketObjPool;
- DWORD m_dwFreeBufferObjPool;
- DWORD m_dwFreeSocketObjHold;
- DWORD m_dwFreeBufferObjHold;
- DWORD m_dwKeepAliveTime;
- DWORD m_dwKeepAliveInterval;
- BOOL m_bMarkSilence;
- private:
- CInitSocket m_wsSocket;
- LPFN_ACCEPTEX m_pfnAcceptEx;
- LPFN_GETACCEPTEXSOCKADDRS m_pfnGetAcceptExSockaddrs;
- LPFN_DISCONNECTEX m_pfnDisconnectEx;
- private:
- ITcpServerListener* m_pListener;
- SOCKET m_soListen;
- HANDLE m_hCompletePort;
- EnServiceState m_enState;
- EnSocketError m_enLastError;
- vector<HANDLE> m_vtWorkerThreads;
- CPrivateHeap m_phSocket;
- CBufferObjPool m_bfObjPool;
- CSpinGuard m_csState;
- TSocketObjPtrPool m_bfActiveSockets;
- TSocketObjPtrList m_lsFreeSocket;
- TSocketObjPtrQueue m_lsGCSocket;
- volatile long m_iRemainAcceptSockets;
- };
|