包含到cocos2d-x裏的tcpsocket源碼

聲明:本文參考了langresser發佈的blog「跨平臺的遊戲客戶端Socket封裝 linux

Socket處理是異步非阻塞的,因此能夠放心的放到主線程處理消息,而且在原做者的基本上進行了系列優化,考慮了客戶端可能創建多個SOCKET,所以加入了Manager概念,與cocos2d-x進行了融合。ios

本文基於cocos2d-x3.0+VS2012c++

點擊下載:network.zipwindows

文件目錄結構截圖:api

 

文件源碼:數組

TCPSocket.h緩存

  1 #ifndef __CC_TCPSOCKET_H__
  2 #define __CC_TCPSOCKET_H__
  3 
  4 #include "cocos2d.h"
  5 #include "ExtensionMacros.h"
  6 #include "WorldPacket.h"
  7 #ifdef WIN32
  8 #include <windows.h>
  9 #include <WinSock.h>
 10 #pragma comment( lib, "ws2_32.lib" )
 11 #else
 12 #include <sys/socket.h>
 13 #include <fcntl.h>
 14 #include <errno.h>
 15 #include <netinet/in.h>
 16 #include <arpa/inet.h>
 17 
 18 #define SOCKET int
 19 #define SOCKET_ERROR -1
 20 #define INVALID_SOCKET -1
 21 
 22 #endif
 23 
 24 NS_CC_EXT_BEGIN
 25 #ifndef CHECKF
 26 #define CHECKF(x) \
 27     do \
 28 { \
 29     if (!(x)) { \
 30     log_msg("CHECKF", #x, __FILE__, __LINE__); \
 31     return 0; \
 32     } \
 33 } while (0)
 34 #endif
 35 
 36 #define _MAX_MSGSIZE 16 * 1024        // 暫定一個消息最大爲16k
 37 #define BLOCKSECONDS    30            // INIT函數阻塞時間
 38 #define INBUFSIZE    (64*1024)        //?    具體尺寸根據剖面報告調整  接收數據的緩存
 39 #define OUTBUFSIZE    (8*1024)        //? 具體尺寸根據剖面報告調整。 發送數據的緩存,當不超過8K時,FLUSH只須要SEND一次
 40 
 41 class CC_DLL TCPSocket
 42 {
 43 public:
 44     TCPSocket(void);
 45     bool    Create(const char* pszServerIP, int nServerPort, int tagid, int nBlockSec = BLOCKSECONDS, bool bKeepAlive = false);
 46     bool    SendMsg(void* pBuf, int nSize);
 47     bool    ReceiveMsg(void* pBuf, int& nSize);
 48     bool    Flush(void);
 49     bool    Check(void);
 50     void    Destroy(void);
 51     SOCKET    GetSocket(void) const { return m_sockClient; }
 52     
 53     int        getTagID(){ return m_tag; }
 54 private:
 55     bool    recvFromSock(void);        // 從網絡中讀取儘量多的數據
 56     bool    hasError();            // 是否發生錯誤,注意,異步模式未完成非錯誤
 57     void    closeSocket();
 58 
 59     SOCKET    m_sockClient;
 60 
 61     // 發送數據緩衝
 62     char    m_bufOutput[OUTBUFSIZE];    //? 可優化爲指針數組
 63     int        m_nOutbufLen;
 64 
 65     // 環形緩衝區
 66     char    m_bufInput[INBUFSIZE];
 67     int        m_nInbufLen;
 68     int        m_nInbufStart;                // INBUF使用循環式隊列,該變量爲隊列起點,0 - (SIZE-1)
 69     int        m_tag;
 70 };
 71 
 72 typedef std::function<bool(int,int,WorldPacket&)> ProAllFunc;    // 接收全部協議,自行處理,@socket標識,@協議頭,@數據包,返回是否分發
 73 typedef std::function<void(int,WorldPacket&)> ProFunc;    // 接收單個協議,@socket標識,@數據包
 74 typedef std::function<void(int)> sckFunc;    // 鏈接成功/斷開事件
 75 
 76 #define SCT_CALLBACK_1(func, _Object) std::bind(&func,_Object, std::placeholders::_1, std::placeholders::_2, std::placeholders::_3)
 77 #define SCT_CALLBACK_2(func, _Object) std::bind(&func,_Object, std::placeholders::_1, std::placeholders::_2)
 78 #define SCT_CALLBACK_3(func, _Object) std::bind(&func,_Object, std::placeholders::_1)
 79 // 建立SOCKET管理器
 80 #define CREATE_TCPSOCKETMGR(pNode)    pNode->addChild(new TCPSocketManager(), 0)
 81 
 82 class CC_DLL TCPSocketManager :
 83     public Node
 84 {
 85 public:
 86     TCPSocketManager()
 87     { 
 88         assert(!mSingleton);
 89         this->mSingleton = this;
 90         // 開啓update
 91         scheduleUpdate(); 
 92     };
 93     ~TCPSocketManager(){};
 94     // 建立socket並添加到管理器
 95     TCPSocket *createSocket(const char* pszServerIP,    // IP地址
 96                             int nServerPort,            // 端口
 97                             int _tag,                    // 標識ID
 98                             int nBlockSec = BLOCKSECONDS, // 阻塞時間ms
 99                             bool bKeepAlive = false);
100     // 註冊協議包
101     void    register_process(const uint16 &entry, ProFunc callback);
102     // 註冊接收全部協議
103     void    register_all_process(ProAllFunc callback){ _pProcess = callback; }
104     // 註冊socket鏈接成功事件
105     void    register_connect(sckFunc callback){ _pOnConnect = callback; }
106     // 註冊socket斷線事件
107     void    register_disconnect(sckFunc callback){ _OnDisconnect = callback; }
108 
109     // 單獨添加socket到管理器
110     bool    addSocket(TCPSocket *pSocket);
111     // 刪除socket
112     bool    removeSocket(int _tag);
113     // 斷開socket
114     void    disconnect(int _tag);
115     // 獲取socket
116     TCPSocket    *GetSocket(int _tag);
117     // 發送消息
118     bool    SendPacket(int _tag, WorldPacket *packet);
119 
120     void    update(float delta);
121 
122     static TCPSocketManager &getSingleton(){ assert(mSingleton); return *mSingleton;}
123 
124 private:    
125     ProAllFunc _pProcess;
126     sckFunc _pOnConnect;
127     sckFunc _OnDisconnect;
128     std::list<TCPSocket*> m_lstSocket;
129     std::map<uint16, ProFunc> _mapProcess;
130     static TCPSocketManager * mSingleton;
131 };
132 
133 #define sSocketMgr    TCPSocketManager::getSingleton()
134 NS_CC_EXT_END
135 
136 #endif //__CC_TCPSOCKET_H__

TCPSocket.cpp服務器

#include "TCPSocket.h"
#include "support/zip_support/unzip.h"


NS_CC_EXT_BEGIN
TCPSocket::TCPSocket()
{ 
    // 初始化
    memset(m_bufOutput, 0, sizeof(m_bufOutput));
    memset(m_bufInput, 0, sizeof(m_bufInput));
}

void TCPSocket::closeSocket()
{
#ifdef WIN32
    closesocket(m_sockClient);
    WSACleanup();
#else
    close(m_sockClient);
#endif
}

bool TCPSocket::Create(const char* pszServerIP, int nServerPort, int tagid, int nBlockSec, bool bKeepAlive /*= FALSE*/)
{
    // 檢查參數
    if(pszServerIP == 0 || strlen(pszServerIP) > 15) {
        return false;
    }

#ifdef WIN32
    WSADATA wsaData;
    WORD version = MAKEWORD(2, 0);
    int ret = WSAStartup(version, &wsaData);//win sock start up
    if (ret != 0) {
        return false;
    }
#endif

    // 建立主套接字
    m_sockClient = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
    if(m_sockClient == INVALID_SOCKET) {
        closeSocket();
        return false;
    }

    // 設置SOCKET爲KEEPALIVE
    if(bKeepAlive)
    {
        int        optval=1;
        if(setsockopt(m_sockClient, SOL_SOCKET, SO_KEEPALIVE, (char *) &optval, sizeof(optval)))
        {
            closeSocket();
            return false;
        }
    }

#ifdef WIN32
    DWORD nMode = 1;
    int nRes = ioctlsocket(m_sockClient, FIONBIO, &nMode);
    if (nRes == SOCKET_ERROR) {
        closeSocket();
        return false;
    }
#else
    // 設置爲非阻塞方式
    fcntl(m_sockClient, F_SETFL, O_NONBLOCK);
#endif

    unsigned long serveraddr = inet_addr(pszServerIP);
    if(serveraddr == INADDR_NONE)    // 檢查IP地址格式錯誤
    {
        closeSocket();
        return false;
    }

    sockaddr_in    addr_in;
    memset((void *)&addr_in, 0, sizeof(addr_in));
    addr_in.sin_family = AF_INET;
    addr_in.sin_port = htons(nServerPort);
    addr_in.sin_addr.s_addr = serveraddr;
    
    if(connect(m_sockClient, (sockaddr *)&addr_in, sizeof(addr_in)) == SOCKET_ERROR) {
        if (hasError()) {
            closeSocket();
            return false;
        }
        else    // WSAWOLDBLOCK
        {
            timeval timeout;
            timeout.tv_sec    = nBlockSec;
            timeout.tv_usec    = 0;
            fd_set writeset, exceptset;
            FD_ZERO(&writeset);
            FD_ZERO(&exceptset);
            FD_SET(m_sockClient, &writeset);
            FD_SET(m_sockClient, &exceptset);

            int ret = select(FD_SETSIZE, NULL, &writeset, &exceptset, &timeout);
            if (ret == 0 || ret < 0) {
                closeSocket();
                return false;
            } else    // ret > 0
            {
                ret = FD_ISSET(m_sockClient, &exceptset);
                if(ret)        // or (!FD_ISSET(m_sockClient, &writeset)
                {
                    closeSocket();
                    return false;
                }
            }
        }
    }

    m_nInbufLen        = 0;
    m_nInbufStart    = 0;
    m_nOutbufLen    = 0;

    struct linger so_linger;
    so_linger.l_onoff = 1;
    so_linger.l_linger = 500;
    setsockopt(m_sockClient, SOL_SOCKET, SO_LINGER, (const char*)&so_linger, sizeof(so_linger));
    m_tag = tagid;
    return true;
}

bool TCPSocket::SendMsg(void* pBuf, int nSize)
{
    if(pBuf == 0 || nSize <= 0) {
        return false;
    }

    if (m_sockClient == INVALID_SOCKET) {
        return false;
    }

    // 檢查通信消息包長度
    int packsize = 0;
    packsize = nSize;

    // 檢測BUF溢出
    if(m_nOutbufLen + nSize > OUTBUFSIZE) {
        // 當即發送OUTBUF中的數據,以清空OUTBUF。
        Flush();
        if(m_nOutbufLen + nSize > OUTBUFSIZE) {
            // 出錯了
            Destroy();
            return false;
        }
    }
    // 數據添加到BUF尾
    memcpy(m_bufOutput + m_nOutbufLen, pBuf, nSize);
    m_nOutbufLen += nSize;
    return true;
}

bool TCPSocket::ReceiveMsg(void* pBuf, int& nSize)
{
    //檢查參數
    if(pBuf == NULL || nSize <= 0) {
        return false;
    }
    
    if (m_sockClient == INVALID_SOCKET) {
        return false;
    }

    // 檢查是否有一個消息(小於2則沒法獲取到消息長度)
    if(m_nInbufLen < 2) {
        //  若是沒有請求成功  或者   若是沒有數據則直接返回
        if(!recvFromSock() || m_nInbufLen < 2) {        // 這個m_nInbufLen更新了
            return false;
        }
    }

    // 計算要拷貝的消息的大小(一個消息,大小爲整個消息的第一個16字節),由於環形緩衝區,因此要分開計算
    int packsize = (unsigned char)m_bufInput[m_nInbufStart+2] +
        (unsigned char)m_bufInput[(m_nInbufStart + 3) % INBUFSIZE] * 256; // 注意字節序,高位+低位

    // 檢測消息包尺寸錯誤 暫定最大16k
    if (packsize <= 0 || packsize > _MAX_MSGSIZE) {
        m_nInbufLen = 0;        // 直接清空INBUF
        m_nInbufStart = 0;
        return false;
    }

    // 檢查消息是否完整(若是將要拷貝的消息大於此時緩衝區數據長度,須要再次請求接收剩餘數據)
    if (packsize > m_nInbufLen) {
        // 若是沒有請求成功   或者    依然沒法獲取到完整的數據包  則返回,直到取得完整包
        if (!recvFromSock() || packsize > m_nInbufLen) {    // 這個m_nInbufLen已更新
            return false;
        }
    }

    // 複製出一個消息
    if(m_nInbufStart + packsize > INBUFSIZE) {
        // 若是一個消息有回捲(被拆成兩份在環形緩衝區的頭尾)
        // 先拷貝環形緩衝區末尾的數據
        int copylen = INBUFSIZE - m_nInbufStart;
        memcpy(pBuf, m_bufInput + m_nInbufStart, copylen);

        // 再拷貝環形緩衝區頭部的剩餘部分
        memcpy((unsigned char *)pBuf + copylen, m_bufInput, packsize - copylen);
        nSize = packsize;
    } else {
        // 消息沒有回捲,能夠一次拷貝出去
        memcpy(pBuf, m_bufInput + m_nInbufStart, packsize);
        nSize = packsize;
    }

    // 從新計算環形緩衝區頭部位置
    m_nInbufStart = (m_nInbufStart + packsize) % INBUFSIZE;
    m_nInbufLen -= packsize;
    return    true;
}

bool TCPSocket::hasError()
{
#ifdef WIN32
    int err = WSAGetLastError();
    if(err != WSAEWOULDBLOCK) {
#else
    int err = errno;
    if(err != EINPROGRESS && err != EAGAIN) {
#endif
        return true;
    }

    return false;
}

// 從網絡中讀取儘量多的數據,實際向服務器請求數據的地方
bool TCPSocket::recvFromSock(void)
{
    if (m_nInbufLen >= INBUFSIZE || m_sockClient == INVALID_SOCKET) {
        return false;
    }

    // 接收第一段數據
    int    savelen, savepos;            // 數據要保存的長度和位置
    if(m_nInbufStart + m_nInbufLen < INBUFSIZE)    {    // INBUF中的剩餘空間有迴繞
        savelen = INBUFSIZE - (m_nInbufStart + m_nInbufLen);        // 後部空間長度,最大接收數據的長度
    } else {
        savelen = INBUFSIZE - m_nInbufLen;
    }

    // 緩衝區數據的末尾
    savepos = (m_nInbufStart + m_nInbufLen) % INBUFSIZE;
    //CHECKF(savepos + savelen <= INBUFSIZE);
    int inlen = recv(m_sockClient, m_bufInput + savepos, savelen, 0);
    if(inlen > 0) {
        // 有接收到數據
        m_nInbufLen += inlen;
        
        if (m_nInbufLen > INBUFSIZE) {
            return false;
        }

        // 接收第二段數據(一次接收沒有完成,接收第二段數據)
        if(inlen == savelen && m_nInbufLen < INBUFSIZE) {
            int savelen = INBUFSIZE - m_nInbufLen;
            int savepos = (m_nInbufStart + m_nInbufLen) % INBUFSIZE;
            //CHECKF(savepos + savelen <= INBUFSIZE);
            inlen = recv(m_sockClient, m_bufInput + savepos, savelen, 0);
            if(inlen > 0) {
                m_nInbufLen += inlen;
                if (m_nInbufLen > INBUFSIZE) {
                    return false;
                }    
            } else if(inlen == 0) {
                Destroy();
                return false;
            } else {
                // 鏈接已斷開或者錯誤(包括阻塞)
                if (hasError()) {
                    Destroy();
                    return false;
                }
            }
        }
    } else if(inlen == 0) {
        Destroy();
        return false;
    } else {
        // 鏈接已斷開或者錯誤(包括阻塞)
        if (hasError()) {
            Destroy();
            return false;
        }
    }

    return true;
}

bool TCPSocket::Flush(void)        //? 若是 OUTBUF > SENDBUF 則須要屢次SEND()
{
    if (m_sockClient == INVALID_SOCKET) {
        return false;
    }

    if(m_nOutbufLen <= 0) {
        return true;
    }
    
    // 發送一段數據
    int    outsize;
    outsize = send(m_sockClient, m_bufOutput, m_nOutbufLen, 0);
    if(outsize > 0) {
        // 刪除已發送的部分
        if(m_nOutbufLen - outsize > 0) {
            memcpy(m_bufOutput, m_bufOutput + outsize, m_nOutbufLen - outsize);
        }

        m_nOutbufLen -= outsize;

        if (m_nOutbufLen < 0) {
            return false;
        }
    } else {
        if (hasError()) {
            Destroy();
            return false;
        }
    }

    return true;
}

bool TCPSocket::Check(void)
{
    // 檢查狀態
    if (m_sockClient == INVALID_SOCKET) {
        return false;
    }

    char buf[1];
    int    ret = recv(m_sockClient, buf, 1, MSG_PEEK);
    if(ret == 0) {
        Destroy();
        return false;
    } else if(ret < 0) {
        if (hasError()) {
            Destroy();
            return false;
        } else {    // 阻塞
            return true;
        }
    } else {    // 有數據
        return true;
    }
    
    return true;
}

void TCPSocket::Destroy(void)
{
    // 關閉
    struct linger so_linger;
    so_linger.l_onoff = 1;
    so_linger.l_linger = 500;
    int ret = setsockopt(m_sockClient, SOL_SOCKET, SO_LINGER, (const char*)&so_linger, sizeof(so_linger));

    closeSocket();

    m_sockClient = INVALID_SOCKET;
    m_nInbufLen = 0;
    m_nInbufStart = 0;
    m_nOutbufLen = 0;

    memset(m_bufOutput, 0, sizeof(m_bufOutput));
    memset(m_bufInput, 0, sizeof(m_bufInput));
}

TCPSocketManager *TCPSocketManager::mSingleton = 0;
TCPSocket *TCPSocketManager::createSocket(const char* pszServerIP,    // IP地址
                            int nServerPort,            // 端口
                            int _tag,                    // 標識ID
                            int nBlockSec,            // 阻塞時間ms
                            bool bKeepAlive)
{
    TCPSocket *pSocket = new TCPSocket();
    if (pSocket->Create(pszServerIP, nServerPort, _tag, nBlockSec, bKeepAlive))
    {
        assert(addSocket(pSocket));
        if (_pOnConnect){
            _pOnConnect(_tag);
        }
        return pSocket;
    }

    delete pSocket;
    return NULL;
}

bool    TCPSocketManager::addSocket(TCPSocket *pSocket)
{
    std::list<TCPSocket*>::iterator iter, enditer = m_lstSocket.end();
    for (iter = m_lstSocket.begin(); iter != enditer; ++ iter)
    {
        if ((*iter)->GetSocket() == pSocket->GetSocket())
            return false;        
    }
    m_lstSocket.push_back(pSocket);
    return true;
}

// 刪除socket
bool    TCPSocketManager::removeSocket(int _tag)
{
    std::list<TCPSocket*>::iterator iter, enditer = m_lstSocket.end();
    for (iter = m_lstSocket.begin(); iter != enditer; ++ iter)
    {
        if ((*iter)->getTagID() == _tag)
        {
            (*iter)->Destroy();
            m_lstSocket.erase(iter);
            return true;
        }
    }
    return false;
}

TCPSocket *TCPSocketManager::GetSocket(int _tag)
{
    std::list<TCPSocket*>::iterator iter, enditer = m_lstSocket.end();
    for (iter = m_lstSocket.begin(); iter != enditer; ++ iter)
    {
        if ((*iter)->getTagID() == _tag)
        {
            return *iter;
        }        
    }
    return NULL;
}

void    TCPSocketManager::update(float delta)
{
    std::list<TCPSocket*>::iterator iter, enditer = m_lstSocket.end();
    for (iter = m_lstSocket.begin(); iter != enditer; ++ iter)
    {
        TCPSocket *pSocket = *iter;
        int _tag = pSocket->getTagID();
        if (!pSocket->Check())
        {// 掉線了
            _OnDisconnect(_tag);
            m_lstSocket.erase(iter);
            break;
        }

        while (true)
        {
            char buffer[_MAX_MSGSIZE] = {0};
            int nSize = sizeof(buffer);
            char *pbufMsg = buffer;
            if (!pSocket->ReceiveMsg(pbufMsg, nSize))
                break;
            WorldPacket packet;
            uint16  _cmd, _length;        
            packet.Write((uint8*)pbufMsg, nSize);
            packet >> _cmd >> _length;
            if (_cmd & 0x8000)
            {
                Byte uncompr[1024] = {0};
                uLong uncomprLen;
                _cmd &= 0x7fff;
                uncompress(uncompr, &uncomprLen, packet.contents()+4, packet.size()-4);
            }

            if (_pProcess == 0 || _pProcess(pSocket->getTagID(), _cmd, packet))
            {// 分發數據                
                std::map<uint16, ProFunc>::iterator mapi = _mapProcess.find(_cmd);
                if (mapi == _mapProcess.end())
                    continue;
                mapi->second(pSocket->getTagID(), packet);
            }
        }
    }
}

void    TCPSocketManager::register_process(const uint16 &entry, ProFunc callback)
{
    _mapProcess[entry] = callback;
}

bool    TCPSocketManager::SendPacket(int _tag, WorldPacket *packet)
{
    std::list<TCPSocket*>::iterator iter, enditer = m_lstSocket.end();
    for (iter = m_lstSocket.begin(); iter != enditer; ++ iter)
    {
        if ((*iter)->getTagID() == _tag)
        {
            (*iter)->SendMsg((void *)packet->contents(), packet->size());
            return (*iter)->Flush();            
        }
    }

    return false;
}

void    TCPSocketManager::disconnect(int _tag)
{
    std::list<TCPSocket*>::iterator iter, enditer = m_lstSocket.end();
    for (iter = m_lstSocket.begin(); iter != enditer; ++ iter)
    {
        if ((*iter)->getTagID() == _tag)
        {
            (*iter)->Destroy();
            if (_OnDisconnect){
                _OnDisconnect(_tag);
            }
            break;
        }
    }
}

NS_CC_EXT_END

ByteBuffer.h  (讀寫數據包)網絡

  1 /****************************************************************************
  2  *
  3  * ByteBuffer Class
  4 *
  5  */
  6 
  7 #ifndef _BYTEBUFFER_H
  8 #define _BYTEBUFFER_H
  9 
 10 #include "Common.h"
 11 //#include "Vector3.h"
 12 
 13 class  ByteBuffer
 14 {
 15 #define DEFAULT_SIZE 0x1000
 16 #define DEFAULT_INCREASE_SIZE 200
 17     uint8 * m_buffer;
 18     size_t m_readPos;
 19     size_t m_writePos;
 20     uint32 m_buffersize;
 21 
 22 public:
 23     /** Creates a bytebuffer with the default size
 24      */
 25     ByteBuffer()
 26     {
 27         m_buffer =0;
 28         m_readPos = m_writePos = 0;
 29         m_buffersize = 0;
 30         reserve(DEFAULT_SIZE);
 31     }
 32 
 33     /** Creates a bytebuffer with the specified size
 34      */
 35     ByteBuffer(size_t res)
 36     {
 37         m_buffer =0;
 38         m_readPos = m_writePos = 0;
 39         m_buffersize = 0;
 40             reserve(res);
 41     }
 42 
 43     /** Frees the allocated buffer
 44      */
 45     ~ByteBuffer()
 46     {
 47         free(m_buffer);
 48     }
 49 
 50 
 51     /** Allocates/reallocates buffer with specified size.
 52      */
 53     void reserve(size_t res)
 54     {
 55         if(m_buffer)
 56             m_buffer = (uint8*)realloc(m_buffer, res);
 57         else
 58             m_buffer = (uint8*)malloc(res);
 59 
 60         m_buffersize = res;
 61     }
 62 
 63 
 64     /** Resets read/write indexes
 65      */
 66     inline void clear()
 67     {
 68         m_readPos = m_writePos = 0;
 69     }
 70 
 71     /** Sets write position
 72      */
 73     inline void resize(size_t size)
 74     {
 75         m_writePos = size;
 76     }
 77 
 78     /** Returns the buffer pointer
 79      */
 80     inline const uint8 * contents()
 81     {
 82         return m_buffer;
 83     }
 84 
 85 
 86     /** Gets the buffer size.
 87      */
 88     uint32 GetBufferSize() { return m_writePos; }
 89 
 90     /** Reads sizeof(T) bytes from the buffer
 91      * @return the bytes read
 92      */
 93     template<typename T>
 94         T Read()
 95     {
 96         if(m_readPos + sizeof(T) > m_writePos)
 97             return (T)0;
 98         T ret = *(T*)&m_buffer[m_readPos];
 99         m_readPos += sizeof(T);
100         return ret;
101     }
102 
103     void skip(size_t len)
104     {
105            if(m_readPos + len > m_writePos)
106                   len = (m_writePos - m_readPos);
107            m_readPos += len;
108     }
109 
110     /** Reads x bytes from the buffer
111      */
112     void read(uint8 * buffer, size_t len)
113     {
114         if(m_readPos + len > m_writePos)
115             len = (m_writePos - m_readPos);
116 
117         memcpy(buffer, &m_buffer[m_readPos], len);
118         m_readPos += len;
119     }
120 
121     /** Writes sizeof(T) bytes to the buffer, while checking for overflows.
122      * @param T data The data to be written
123      */
124     template<typename T>
125         void Write(const T & data)
126     {
127         size_t  new_size = m_writePos + sizeof(T);
128         if(new_size > m_buffersize)
129         {
130             new_size = (new_size / DEFAULT_INCREASE_SIZE + 1) * DEFAULT_INCREASE_SIZE;
131             reserve(new_size);
132         }
133 
134         *(T*)&m_buffer[m_writePos] = data;
135         m_writePos += sizeof(T);
136     }
137 
138     /** writes x bytes to the buffer, while checking for overflows
139      * @param ptr the data to be written
140      * @param size byte count
141     */
142     void Write(const uint8 * data, size_t size)
143     {
144         size_t  new_size = m_writePos + size;
145                 if(new_size > m_buffersize)
146                 {
147                         new_size = (new_size / DEFAULT_INCREASE_SIZE + 1) * DEFAULT_INCREASE_SIZE;
148                         reserve(new_size);
149                 }
150 
151         memcpy(&m_buffer[m_writePos], data, size);
152         m_writePos += size;
153     }
154 
155     /** Ensures the buffer is big enough to fit the specified number of bytes.
156      * @param bytes number of bytes to fit
157      */
158     inline void EnsureBufferSize(uint32 Bytes)
159     {
160         size_t  new_size = m_writePos + Bytes;
161         if(new_size > m_buffersize)
162                 {
163                         new_size = (new_size / DEFAULT_INCREASE_SIZE + 1) * DEFAULT_INCREASE_SIZE;
164                         reserve(new_size);
165                 }
166 
167     }
168 
169     /** These are the default read/write operators.
170      */
171 #define DEFINE_BUFFER_READ_OPERATOR(type) void operator >> (type& dest) { dest = Read<type>(); }
172 #define DEFINE_BUFFER_WRITE_OPERATOR(type) void operator << (const type src) { Write<type>(src); }
173 
174     /** Fast read/write operators without using the templated read/write functions.
175      */
176 #define DEFINE_FAST_READ_OPERATOR(type, size) ByteBuffer& operator >> (type& dest) { if(m_readPos + size > m_writePos) { dest = (type)0; return *this; } else { dest = *(type*)&m_buffer[m_readPos]; m_readPos += size; return *this; } }
177 #define DEFINE_FAST_WRITE_OPERATOR(type, size) ByteBuffer& operator << (const type src) { if(m_writePos + size > m_buffersize) { reserve(m_buffersize + DEFAULT_INCREASE_SIZE); } *(type*)&m_buffer[m_writePos] = src; m_writePos += size; return *this; }
178 
179     /** Integer/float r/w operators
180     */
181     DEFINE_FAST_READ_OPERATOR(uint64, 8);
182     DEFINE_FAST_READ_OPERATOR(uint32, 4);
183     DEFINE_FAST_READ_OPERATOR(uint16, 2);
184     DEFINE_FAST_READ_OPERATOR(uint8, 1);
185     DEFINE_FAST_READ_OPERATOR(int64, 8);
186     DEFINE_FAST_READ_OPERATOR(int32, 4);
187     DEFINE_FAST_READ_OPERATOR(int16, 2);
188     DEFINE_FAST_READ_OPERATOR(int8, 1);
189     DEFINE_FAST_READ_OPERATOR(float, 4);
190     DEFINE_FAST_READ_OPERATOR(double, 8);
191 
192     DEFINE_FAST_WRITE_OPERATOR(uint64, 8);
193     DEFINE_FAST_WRITE_OPERATOR(uint32, 4);
194     DEFINE_FAST_WRITE_OPERATOR(uint16, 2);
195     DEFINE_FAST_WRITE_OPERATOR(uint8, 1);
196     DEFINE_FAST_WRITE_OPERATOR(int64, 8);
197     DEFINE_FAST_WRITE_OPERATOR(int32, 4);
198     DEFINE_FAST_WRITE_OPERATOR(int16, 2);
199     DEFINE_FAST_WRITE_OPERATOR(int8, 1);
200     DEFINE_FAST_WRITE_OPERATOR(float, 4);
201     DEFINE_FAST_WRITE_OPERATOR(double, 8);
202 
203     /** boolean (1-byte) read/write operators
204      */
205     DEFINE_FAST_WRITE_OPERATOR(bool, 1);
206     ByteBuffer& operator >> (bool & dst) { dst = (Read<char>() > 0 ? true : false); return *this; }
207 
208     /** string (null-terminated) operators
209      */
210     ByteBuffer& operator << (const std::string & value) { EnsureBufferSize(value.length() + 1); memcpy(&m_buffer[m_writePos], value.c_str(), value.length()+1); m_writePos += (value.length() + 1); return *this; }
211     ByteBuffer& operator >> (std::string & dest)
212     {
213         dest.clear();
214         char c;
215         for(;;)
216         {
217             c = Read<char>();
218             if(c == 0) break;
219             dest += c;
220         }
221         return *this;
222     }
223 
224     /** Gets the write position
225      * @return buffer size
226      */
227     inline size_t size() { return m_writePos; }
228 
229     /** read/write position setting/getting
230      */
231     inline size_t rpos() { return m_readPos; }
232     inline size_t wpos() { return m_writePos; }
233     inline void rpos(size_t p) { assert(p <= m_writePos); m_readPos = p; }
234     inline void wpos(size_t p) { assert(p <= m_buffersize); m_writePos = p; }
235 
236     template<typename T> size_t writeVector(std::vector<T> &v)
237     {
238             for (typename std::vector<T>::const_iterator i = v.begin(); i != v.end(); i++) {
239                     Write<T>(*i);
240             }
241             return v.size();
242         
243     }
244     template<typename T> size_t readVector(size_t vsize, std::vector<T> &v)
245     {
246         
247         v.clear();
248         while(vsize--) {
249             T t = Read<T>();
250             v.push_back(t);
251         }
252         return v.size();
253     }
254 
255     template<typename T> size_t writeList(std::list<T> &v)
256     {
257             for (typename std::list<T>::const_iterator i = v.begin(); i != v.end(); i++) {
258                     Write<T>(*i);
259             }
260             return v.size();
261         
262     }
263     template<typename T> size_t readList(size_t vsize, std::list<T> &v)
264     {
265         
266         v.clear();
267         while(vsize--) {
268             T t = Read<T>();
269             v.push_back(t);
270         }
271         return v.size();
272     }
273 
274     template <typename K, typename V> size_t writeMap(const std::map<K, V> &m)
275     {
276         for (typename std::map<K, V>::const_iterator i = m.begin(); i != m.end(); i++) {
277             Write<K>(i->first);
278             Write<V>(i->second);
279         }
280         return m.size();
281     }
282 
283     template <typename K, typename V> size_t readMap(size_t msize, std::map<K, V> &m)
284     {
285         m.clear();
286         while(msize--) {
287             K k = Read<K>();
288             V v = Read<V>();
289             m.insert(make_pair(k, v));
290         }
291         return m.size();
292     }
293 
294 };
295 
296 ///////////////////////////////////////////////////////////////////////////////
297 ///////////////////////////////////////////////////////////////////////////////
298 ///////////////////////////////////////////////////////////////////////////////
299 
300 template <typename T> ByteBuffer &operator<<(ByteBuffer &b, const std::vector<T> & v)
301 {
302     b << (uint32)v.size();
303     for (typename std::vector<T>::const_iterator i = v.begin(); i != v.end(); i++) {
304         b << *i;
305     }
306     return b;
307 }
308 
309 template <typename T> ByteBuffer &operator>>(ByteBuffer &b, std::vector<T> &v)
310 {
311     uint32 vsize;
312     b >> vsize;
313     v.clear();
314     while(vsize--) {
315         T t;
316         b >> t;
317         v.push_back(t);
318     }
319     return b;
320 }
321 
322 template <typename T> ByteBuffer &operator<<(ByteBuffer &b, const std::list<T> & v)
323 {
324     b << (uint32)v.size();
325     for (typename std::list<T>::const_iterator i = v.begin(); i != v.end(); i++) {
326         b << *i;
327     }
328     return b;
329 }
330 
331 template <typename T> ByteBuffer &operator>>(ByteBuffer &b, std::list<T> &v)
332 {
333     uint32 vsize;
334     b >> vsize;
335     v.clear();
336     while(vsize--) {
337         T t;
338         b >> t;
339         v.push_back(t);
340     }
341     return b;
342 }
343 
344 template <typename K, typename V> ByteBuffer &operator<<(ByteBuffer &b, const std::map<K, V> &m)
345 {
346     b << (uint32)m.size();
347     for (typename std::map<K, V>::const_iterator i = m.begin(); i != m.end(); i++) {
348         b << i->first << i->second;
349     }
350     return b;
351 }
352 
353 template <typename K, typename V> ByteBuffer &operator>>(ByteBuffer &b, std::map<K, V> &m)
354 {
355     uint32 msize;
356     b >> msize;
357     m.clear();
358     while(msize--) {
359         K k;
360         V v;
361         b >> k >> v;
362         m.insert(make_pair(k, v));
363     }
364     return b;
365 }
366 
367 #endif

WorldPacket.h  (讀寫擴展類,協議前2字節表示協議包,其次2字節表示數據大小)異步

  1 #ifndef WGSERVER_WORLDPACKET_H
  2 #define WGSERVER_WORLDPACKET_H
  3 
  4 
  5 #include "ByteBuffer.h"
  6 
  7 
  8 class WorldPacket : public ByteBuffer
  9 {
 10 public:
 11     __inline WorldPacket() : ByteBuffer(), m_opcode(0) { }
 12     __inline WorldPacket(uint16 opcode, size_t res) : ByteBuffer(res), m_opcode(opcode) {}
 13     __inline WorldPacket(size_t res) : ByteBuffer(res), m_opcode(0) { }
 14     __inline WorldPacket(const WorldPacket &packet) : ByteBuffer(packet), m_opcode(packet.m_opcode) {}
 15 
 16     //! Clear packet and set opcode all in one mighty blow
 17     __inline void Initialize(uint16 opcode )
 18     {
 19         clear();
 20         m_opcode = opcode;
 21     }
 22 
 23     __inline uint16 GetOpcode() const { return m_opcode; }
 24     __inline void SetOpcode(const uint16 & opcode) { m_opcode = opcode; }
 25     __inline void SetLength(const uint16 & len)    { 
 26         uint16 * plen = (uint16 * ) &(contents()[2]);
 27         *plen = len;
 28     }
 29 
 30     __inline std::string getString()
 31     {
 32         //std::string buff = 
 33         return (const char*)contents();
 34     }
 35     
 36  template<typename T>
 37        void SetOffset(const uint16 & offset, const T value ) {
 38         T * pval = (T *) &(contents()[offset]);
 39         *pval = value;
 40     }
 41 
 42 public:
 43     /**
 44      * @建立時間 2011-08-31
 45      * @建立人 李志勇
 46      * @函數做用 向數據包追加字符串
 47      * @參數
 48      *     @packet 數據封包指針
 49      *  @str 追加的字符串
 50      */
 51     template<typename T> void AppendPacketString(std::string str)
 52     {
 53         T ilen = (T)str.size();
 54         *this << ilen;
 55         if (ilen > 0)
 56             Write((const uint8 *) str.c_str(), ilen);
 57     }
 58     /**
 59      * @建立時間 2011-08-31
 60      * @建立人 李志勇
 61      * @函數做用 獲取字符串
 62      * @參數
 63      *     @packet 數據封包
 64      * @返回 是否成功
 65      */
 66     template<typename T> bool GetPacketString(std::string &str)
 67     {
 68         T ilen;
 69         *this >> ilen;
 70         if (ilen == 0)
 71             return true;
 72         uint8 buf[ilen+1];
 73         uint16 plen = size();
 74         if (plen < ilen)
 75             return false;
 76         read(buf, ilen);
 77         buf[ilen] = 0;
 78         str = (char*)buf;
 79         return true;
 80     }
 81 
 82 
 83 protected:
 84     uint16 m_opcode;
 85 };
 86 
 87 /*
 88 template<uint32 Size>
 89 class SERVER_DECL StackWorldPacket : public StackBuffer<Size>
 90 {
 91     uint16 m_opcode;
 92 public:
 93     __inline StackWorldPacket(uint16 opcode) : StackBuffer<Size>(), m_opcode(opcode) { }
 94 
 95     //! Clear packet and set opcode all in one mighty blow
 96     __inline void Initialize(uint16 opcode )
 97     {
 98         StackBuffer<Size>::Clear();
 99         m_opcode = opcode;
100     }
101 
102     uint16 GetOpcode() { return m_opcode; }
103     __inline void SetOpcode(uint16 opcode) { m_opcode = opcode; }
104 };
105 */
106 
107 #endif

Common.h (bytebuffer.h所需的一些定義,該文件來自「魔獸世界服務端」源碼,cocos2d-x自身有個CCommon.h文件,若是把兩個文件合併的話,在linux下編譯會發生衝突,衝突較多,本人也沒有作區分,所以建議獨立出來)

  1 #ifndef WGSERVER_COMMON_H
  2 #define WGSERVER_COMMON_H
  3 
  4 
  5 #ifdef WIN32
  6 #pragma warning(disable:4996)
  7 #define _CRT_SECURE_NO_DEPRECATE 1
  8 #define _CRT_SECURE_COPP_OVERLOAD_STANDARD_NAMES 1
  9 #pragma warning(disable:4251)        // dll-interface bullshit
 10 #endif
 11 
 12 enum TimeVariables
 13 {
 14     TIME_SECOND = 1,
 15     TIME_MINUTE = TIME_SECOND * 60,
 16     TIME_HOUR   = TIME_MINUTE * 60,
 17     TIME_DAY    = TIME_HOUR * 24,
 18     TIME_MONTH    = TIME_DAY * 30,
 19     TIME_YEAR    = TIME_MONTH * 12
 20 };
 21 
 22 enum MsTimeVariables
 23 {
 24     MSTIME_SECOND = 1000,
 25     MSTIME_MINUTE = MSTIME_SECOND * 60,
 26     MSTIME_HOUR   = MSTIME_MINUTE * 60,
 27     MSTIME_DAY      = MSTIME_HOUR * 24
 28 };
 29 
 30 #ifdef WIN32
 31 #define WGSERVER_INLINE __forceinline
 32 #else
 33 #define WGSERVER_INLINE inline
 34 #endif
 35 
 36 #ifdef HAVE_CONFIG_H
 37 # include <config.h>
 38 #endif
 39 
 40 
 41 /* Define this if you're using a big-endian machine */
 42 #ifdef USING_BIG_ENDIAN
 43 #include <machine/byte_order.h>
 44 #define bswap_16(x) NXSwapShort(x)
 45 #define bswap_32(x) NXSwapInt(x)
 46 #define bswap_64(x) NXSwapLongLong(x)
 47 #endif
 48 
 49 #include <cstdlib>
 50 #include <stdio.h>
 51 #include <stdlib.h>
 52 #include <stdarg.h>
 53 #include <time.h>
 54 #include <math.h>
 55 #include <errno.h>
 56 //#include <sys/syscall.h>
 57 
 58 
 59 #if defined( __WIN32__ ) || defined( WIN32 ) || defined( _WIN32 )
 60 #  define WIN32_LEAN_AND_MEAN
 61 //#  define _WIN32_WINNT 0x0500
 62 #  define NOMINMAX
 63 #  include <windows.h>
 64 #else
 65 #  include <string.h>
 66 #  define MAX_PATH 1024
 67 #endif
 68 
 69 #ifdef min
 70 #undef min
 71 #endif
 72 
 73 #ifdef max
 74 #undef max
 75 #endif
 76 
 77 #ifdef CONFIG_USE_SELECT
 78 #undef FD_SETSIZE
 79 #define FD_SETSIZE 2048 
 80 #endif
 81 
 82 #if defined( __WIN32__ ) || defined( WIN32 ) || defined( _WIN32 )
 83 #include <winsock2.h>
 84 #include <ws2tcpip.h>
 85 #else
 86 #include <sys/time.h>
 87 #include <sys/types.h>
 88 #include <sys/ioctl.h>
 89 #include <sys/socket.h>
 90 #include <netinet/in.h>
 91 #include <arpa/inet.h>
 92 #include <unistd.h>
 93 #include <signal.h>
 94 #include <netdb.h>
 95 #endif
 96 
 97 // current platform and compiler
 98 #define PLATFORM_WIN32 0
 99 #define PLATFORM_UNIX  1
100 #define PLATFORM_APPLE 2
101 
102 #define UNIX_FLAVOUR_LINUX 1
103 #define UNIX_FLAVOUR_BSD 2
104 #define UNIX_FLAVOUR_OTHER 3
105 #define UNIX_FLAVOUR_OSX 4
106 
107 #if defined( __WIN32__ ) || defined( WIN32 ) || defined( _WIN32 )
108 #  define PLATFORM PLATFORM_WIN32
109 #elif defined( __APPLE_CC__ )
110 #  define PLATFORM PLATFORM_APPLE
111 #else
112 #  define PLATFORM PLATFORM_UNIX
113 #endif
114 
115 #define COMPILER_MICROSOFT 0
116 #define COMPILER_GNU       1
117 #define COMPILER_BORLAND   2
118 
119 #ifdef _MSC_VER
120 #  define COMPILER COMPILER_MICROSOFT
121 #elif defined( __BORLANDC__ )
122 #  define COMPILER COMPILER_BORLAND
123 #elif defined( __GNUC__ )
124 #  define COMPILER COMPILER_GNU
125 #else
126 #  pragma error "FATAL ERROR: Unknown compiler."
127 #endif
128 
129 #if PLATFORM == PLATFORM_UNIX || PLATFORM == PLATFORM_APPLE
130 #ifdef HAVE_DARWIN
131 #define PLATFORM_TEXT "MacOSX"
132 #define UNIX_FLAVOUR UNIX_FLAVOUR_OSX
133 #else
134 #ifdef USE_KQUEUE
135 #define PLATFORM_TEXT "FreeBSD"
136 #define UNIX_FLAVOUR UNIX_FLAVOUR_BSD
137 #else
138 #define PLATFORM_TEXT "Linux"
139 #define UNIX_FLAVOUR UNIX_FLAVOUR_LINUX
140 #endif
141 #endif
142 #endif
143 
144 #if PLATFORM == PLATFORM_WIN32
145 #define PLATFORM_TEXT "Win32"
146 #endif
147 
148 #ifdef _DEBUG
149 #define CONFIG "Debug"
150 #else
151 #define CONFIG "Release"
152 #endif
153 
154 #ifdef USING_BIG_ENDIAN
155 #define ARCH "PPC"
156 #else
157 #ifdef X64
158 #define ARCH "X64"
159 #else
160 #define ARCH "X86"
161 #endif
162 #endif
163 
164 /*#if COMPILER == COMPILER_MICROSOFT
165 #  pragma warning( disable : 4267 ) // conversion from 'size_t' to 'int', possible loss of data
166 #  pragma warning( disable : 4311 ) // 'type cast': pointer truncation from HMODULE to uint32
167 #  pragma warning( disable : 4786 ) // identifier was truncated to '255' characters in the debug information
168 #  pragma warning( disable : 4146 )
169 #  pragma warning( disable : 4800 )
170 #endif*/
171 
172 #if PLATFORM == PLATFORM_WIN32
173 #define STRCASECMP stricmp
174 #else
175 #define STRCASECMP strcasecmp
176 #endif
177 
178 #if PLATFORM == PLATFORM_WIN32
179     #define ASYNC_NET
180 #endif
181 
182 #ifdef USE_EPOLL
183     #define CONFIG_USE_EPOLL
184 #endif
185 #ifdef USE_KQUEUE
186     #define CONFIG_USE_KQUEUE
187 #endif
188 #ifdef USE_SELECT
189     #define CONFIG_USE_SELECT
190 #endif
191 #ifdef USE_POLL
192     #define CONFIG_USE_POLL
193 #endif
194 
195 #ifdef min
196 #undef min
197 #endif
198 
199 #ifdef max
200 #undef max
201 #endif
202 
203 #include <cstdlib>
204 #include <set>
205 #include <list>
206 #include <string>
207 #include <map>
208 #include <queue>
209 #include <sstream>
210 #include <algorithm>
211 #include <cstring>
212 #include <climits>
213 #include <cstdlib>
214 //#include <iostream>
215 
216 #if defined ( __GNUC__ )
217 #    define LIKELY( _x ) \
218         __builtin_expect( ( _x ), 1 )
219 #    define UNLIKELY( _x ) \
220          __builtin_expect( ( _x ), 0 )
221 #else
222 #    define LIKELY( _x ) \
223         _x
224 #    define UNLIKELY( _x ) \
225         _x
226 #endif
227 
228 #if defined (__GNUC__)
229 #  define GCC_VERSION (__GNUC__ * 10000 \
230                        + __GNUC_MINOR__ * 100 \
231                        + __GNUC_PATCHLEVEL__)
232 #endif
233 
234 
235 #ifndef WIN32
236 #ifndef X64
237 #  if defined (__GNUC__)
238 #    if GCC_VERSION >= 30400
239 #         ifdef HAVE_DARWIN
240 #          define __fastcall
241 #         else
242 #              define __fastcall __attribute__((__fastcall__))
243 #         endif
244 #    else
245 #      define __fastcall __attribute__((__regparm__(3)))
246 #    endif
247 #  else
248 #    define __fastcall __attribute__((__fastcall__))
249 #  endif
250 #else
251 #define __fastcall  
252 #endif
253 #endif
254 
255 #ifdef HAVE_STDCXX_0X
256 #include <unordered_map>
257 #include <unordered_set>
258 #elif COMPILER == COMPILER_GNU && __GNUC__ >= 3
259 #include <ext/hash_map>
260 #include <ext/hash_set>
261 #else
262 #include <hash_map>
263 #include <hash_set>
264 #endif
265 
266 
267 
268 #ifdef _STLPORT_VERSION
269 #define HM_NAMESPACE std
270 using std::hash_map;
271 using std::hash_set;
272 #elif COMPILER == COMPILER_MICROSOFT && _MSC_VER >= 1300
273 #define HM_NAMESPACE stdext
274 using stdext::hash_map;
275 using stdext::hash_set;
276 #define ENABLE_SHITTY_STL_HACKS 1
277 
278 /*
279         #   if GNUC_COMP_VER >= 430
280         #       define HashMap ::std::tr1::unordered_map
281         #               define HashSet ::std::tr1::unordered_set
282         #    else
283         #       define HashMap ::__gnu_cxx::hash_map
284         #       define HashSet ::__gnu_cxx::hash_set
285         #    endif
286 */
287 
288 #define HashMap ::__gnu_cxx::hash_map
289 #define HashSet ::__gnu_cxx::hash_set
290 
291 
292 // hacky stuff for vc++
293 #define snprintf _snprintf
294 #define vsnprintf _vsnprintf
295 //#define strlen lstrlen
296 
297 #ifdef WIN32
298 //typedef char TCHAR;
299 #define __T(x) x
300 #endif
301 
302 // cebernic added it
303 #define __utf8(x) _StringToUTF8(x)
304 #define __ansi(x) _StringToANSI(x)
305 #define __isutf8(x) _IsStringUTF8(x)
306 
307 
308 #elif COMPILER == COMPILER_INTEL
309 #define HM_NAMESPACE std
310 using std::hash_map;
311 using std::hash_set;
312 #elif defined(HAVE_STDCXX_0X)
313 #define HM_NAMESPACE std
314 #define hash_map unordered_map
315 #define hash_set unordered_set
316 using std::unordered_map;
317 using std::unordered_set;
318 #elif COMPILER == COMPILER_GNU && __GNUC__ >= 3
319 #define HM_NAMESPACE __gnu_cxx
320 using __gnu_cxx::hash_map;
321 using __gnu_cxx::hash_set;
322 
323 namespace __gnu_cxx
324 {
325     template<> struct hash<unsigned long long>
326     {
327         size_t operator()(const unsigned long long &__x) const { return (size_t)__x; }
328     };
329     template<typename T> struct hash<T *>
330     {
331         size_t operator()(T * const &__x) const { return (size_t)__x; }
332     };
333 
334 };
335 
336 #else
337 #define HM_NAMESPACE std
338 using std::hash_map;
339 #endif
340 
341 /* Use correct types for x64 platforms, too */
342 #if COMPILER != COMPILER_GNU
343 typedef signed __int64 int64;
344 typedef signed __int32 int32;
345 typedef signed __int16 int16;
346 typedef signed __int8 int8;
347 
348 typedef unsigned __int64 uint64;
349 typedef unsigned __int32 uint32;
350 typedef unsigned __int16 uint16;
351 typedef unsigned __int8 uint8;
352 typedef float    Real;
353 #else
354 
355 typedef int64_t int64;
356 typedef int32_t int32;
357 typedef int16_t int16;
358 typedef int8_t int8;
359 typedef uint64_t uint64;
360 typedef uint32_t uint32;
361 typedef uint16_t uint16;
362 typedef uint8_t uint8;
363 typedef uint32_t DWORD;
364 typedef float    Real;
365 
366 #endif
367 
368 /* these can be optimized into assembly */
369 #ifdef USING_BIG_ENDIAN
370 
371 WGSERVER_INLINE static void swap16(uint16* p) { *p = bswap_16((uint16_t)*p); }
372 WGSERVER_INLINE static void swap32(uint32* p) { *p = bswap_32((uint32_t)*p); }
373 WGSERVER_INLINE static void swap64(uint64* p) { *p = bswap_64((uint64_t)*p);; }
374 
375 WGSERVER_INLINE static float swapfloat(float p)
376 {
377     union { float asfloat; uint8 asbytes[4]; } u1, u2;
378     u1.asfloat = p;
379     /* swap! */
380     u2.asbytes[0] = u1.asbytes[3];
381     u2.asbytes[1] = u1.asbytes[2];
382     u2.asbytes[2] = u1.asbytes[1];
383     u2.asbytes[3] = u1.asbytes[0];
384     
385     return u2.asfloat;
386 }
387 
388 WGSERVER_INLINE static double swapdouble(double p)
389 {
390     union { double asfloat; uint8 asbytes[8]; } u1, u2;
391     u1.asfloat = p;
392     /* swap! */
393     u2.asbytes[0] = u1.asbytes[7];
394     u2.asbytes[1] = u1.asbytes[6];
395     u2.asbytes[2] = u1.asbytes[5];
396     u2.asbytes[3] = u1.asbytes[4];
397     u2.asbytes[4] = u1.asbytes[3];
398     u2.asbytes[5] = u1.asbytes[2];
399     u2.asbytes[6] = u1.asbytes[1];
400     u2.asbytes[7] = u1.asbytes[0];
401 
402     return u2.asfloat;
403 }
404 
405 WGSERVER_INLINE static void swapfloat(float * p)
406 {
407     union { float asfloat; uint8 asbytes[4]; } u1, u2;
408     u1.asfloat = *p;
409     /* swap! */
410     u2.asbytes[0] = u1.asbytes[3];
411     u2.asbytes[1] = u1.asbytes[2];
412     u2.asbytes[2] = u1.asbytes[1];
413     u2.asbytes[3] = u1.asbytes[0];
414     *p = u2.asfloat;
415 }
416 
417 WGSERVER_INLINE static void swapdouble(double * p)
418 {
419     union { double asfloat; uint8 asbytes[8]; } u1, u2;
420     u1.asfloat = *p;
421     /* swap! */
422     u2.asbytes[0] = u1.asbytes[7];
423     u2.asbytes[1] = u1.asbytes[6];
424     u2.asbytes[2] = u1.asbytes[5];
425     u2.asbytes[3] = u1.asbytes[4];
426     u2.asbytes[4] = u1.asbytes[3];
427     u2.asbytes[5] = u1.asbytes[2];
428     u2.asbytes[6] = u1.asbytes[1];
429     u2.asbytes[7] = u1.asbytes[0];
430     *p = u2.asfloat;
431 }
432 
433 WGSERVER_INLINE static uint16 swap16(uint16 p) { return bswap_16((uint16_t)p); }
434 WGSERVER_INLINE static uint32 swap32(uint32 p) { return bswap_32((uint32_t)p); }
435 WGSERVER_INLINE static uint64 swap64(uint64 p)  { return bswap_64((uint64_t)p); }
436 
437 WGSERVER_INLINE static void swap16(int16* p) { *p = bswap_16((uint16_t)*p); }
438 WGSERVER_INLINE static void swap32(int32* p) { *p = bswap_32((uint32_t)*p); }
439 WGSERVER_INLINE static void swap64(int64* p) { *p = bswap_64((uint64_t)*p); }
440 
441 WGSERVER_INLINE static int16 swap16(int16 p) { return bswap_16((uint16_t)p); }
442 WGSERVER_INLINE static int32 swap32(int32 p) { return bswap_32((uint32_t)p); }
443 WGSERVER_INLINE static int64 swap64(int64 p)  { return bswap_64((uint64_t)p); }
444 
445 #endif
446 /* 
447 Scripting system exports/imports
448 */
449 
450 #ifdef WIN32
451     #ifndef SCRIPTLIB
452         #define SERVER_DECL __declspec(dllexport)
453         #define SCRIPT_DECL __declspec(dllimport)
454     #else
455         #define SERVER_DECL __declspec(dllimport)
456         #define SCRIPT_DECL __declspec(dllexport)
457     #endif
458 #else
459     #define SERVER_DECL 
460     #define SCRIPT_DECL 
461 #endif
462 
463 // Include all threading files
464 #include <assert.h>
465 //#include "Threading/Threading.h"
466 
467 //#include "MersenneTwister.h"
468 
469 #if COMPILER == COMPILER_MICROSOFT
470 
471 #define I64FMT "%016I64X"
472 #define I64FMTD "%I64u"
473 #define SI64FMTD "%I64d"
474 #define snprintf _snprintf
475 #define atoll __atoi64
476 
477 #else
478 
479 #define stricmp strcasecmp
480 #define strnicmp strncasecmp
481 #define I64FMT "%016llX"
482 #define I64FMTD "%llu"
483 #define SI64FMTD "%lld"
484 
485 #endif
486 
487 #ifndef WIN32
488 #ifdef USING_BIG_ENDIAN
489 //#define GUID_HIPART(x) (*((uint32*)&(x)))
490 //#define GUID_LOPART(x) (*(((uint32*)&(x))+1))
491 #define GUID_LOPART(x) ( ( x >> 32 ) )
492 #define GUID_HIPART(x) ( ( x & 0x00000000ffffffff ) )
493 #else
494 #define GUID_HIPART(x) ( ( x >> 32 ) )
495 #define GUID_LOPART(x) ( ( x & 0x00000000ffffffff ) )
496 #endif
497 #else
498 #define GUID_HIPART(x) (*(((uint32*)&(x))+1))
499 #define GUID_LOPART(x) (*((uint32*)&(x)))
500 #endif
501 
502 #define atol(a) strtoul( a, NULL, 10)
503 
504 #define STRINGIZE(a) #a
505 
506 // fix buggy MSVC's for variable scoping to be reliable =S
507 #define for if(true) for
508 
509 #if COMPILER == COMPILER_MICROSOFT && _MSC_VER >= 1400
510 #pragma float_control(push)
511 #pragma float_control(precise, on)
512 #endif
513 
514 // fast int abs
515 static inline int int32abs( const int value )
516 {
517     return (value ^ (value >> 31)) - (value >> 31);
518 }
519 
520 // fast int abs and recast to unsigned
521 static inline uint32 int32abs2uint32( const int value )
522 {
523     return (uint32)(value ^ (value >> 31)) - (value >> 31);
524 }
525 
526 /// Fastest Method of float2int32
527 static inline int float2int32(const float value)
528 {
529 #if !defined(X64) && COMPILER == COMPILER_MICROSOFT && !defined(USING_BIG_ENDIAN)
530     int i;
531     __asm {
532         fld value
533         frndint
534         fistp i
535     }
536     return i;
537 #else
538     union { int asInt[2]; double asDouble; } n;
539     n.asDouble = value + 6755399441055744.0;
540 
541 #if USING_BIG_ENDIAN
542     return n.asInt [1];
543 #else
544     return n.asInt [0];
545 #endif
546 #endif
547 }
548 
549 /// Fastest Method of long2int32
550 static inline int long2int32(const double value)
551 {
552 #if !defined(X64) && COMPILER == COMPILER_MICROSOFT && !defined(USING_BIG_ENDIAN)
553     int i;
554     __asm {
555         fld value
556         frndint
557         fistp i
558     }
559     return i;
560 #else
561   union { int asInt[2]; double asDouble; } n;
562   n.asDouble = value + 6755399441055744.0;
563 
564 #if USING_BIG_ENDIAN
565   return n.asInt [1];
566 #else
567   return n.asInt [0];
568 #endif
569 #endif
570 }
571 
572 #if COMPILER == COMPILER_MICROSOFT && _MSC_VER >= 1400
573 #pragma float_control(pop)
574 #endif
575 
576 #ifndef WIN32
577 #include <sys/timeb.h>
578 #endif
579 
580 
581 extern uint32 system_start_time_t;
582 
583 WGSERVER_INLINE uint32 now()
584 {    
585 #ifdef WIN32
586     return GetTickCount();
587 #else
588     struct timeval tv;
589     gettimeofday(&tv, NULL);
590     return ((tv.tv_sec - system_start_time_t) * 1000) + (tv.tv_usec / 1000);
591 #endif
592 }
593 
594 #ifndef WIN32
595 #define FALSE   0
596 #define TRUE    1
597 #endif
598 
599 #ifndef WIN32
600 #define Sleep(ms) usleep(1000*ms)
601 #endif
602 
603 /*#ifdef WIN32
604 #ifndef __SHOW_STUPID_WARNINGS__
605 #pragma warning(disable:4018)
606 #pragma warning(disable:4244)
607 #pragma warning(disable:4305) 
608 #pragma warning(disable:4748)
609 #pragma warning(disable:4800) 
610 #pragma warning(disable:4996)
611 #pragma warning(disable:4251)
612 #endif      
613 #endif
614 
615 #undef INTEL_COMPILER
616 #ifdef INTEL_COMPILER
617 #pragma warning(disable:279)
618 #pragma warning(disable:1744)
619 #pragma warning(disable:1740)
620 #endif*/
621 
622 //#include "Util.h"
623 struct WayPoint
624 {
625     WayPoint()
626     {
627         o = 0.0f;
628     }
629     uint32 id;
630     float x;
631     float y;
632     float z;
633     float o;
634     uint32 waittime; //ms
635     uint32 flags;
636     bool forwardemoteoneshot;
637     uint32 forwardemoteid;
638     bool backwardemoteoneshot;
639     uint32 backwardemoteid;
640     uint32 forwardskinid;
641     uint32 backwardskinid;
642 
643 };
644 
645 struct spawn_timed_emotes
646 {
647     uint8        type; //1 standstate, 2 emotestate, 3 emoteoneshot
648     uint32        value; //get yar list elsewhere
649     char        *msg; //maybe we wish to say smething while changing emote state
650     uint8        msg_type; //yell ? say ?
651     uint8        msg_lang; //yell ? say ?
652     uint32        expire_after; //going to nex faze in
653 };
654 typedef std::list<spawn_timed_emotes*> TimedEmoteList;
655 
656 WGSERVER_INLINE void reverse_array(uint8 * pointer, size_t count)
657 {
658     size_t x;
659     uint8 * temp = (uint8*)malloc(count);
660     memcpy(temp, pointer, count);
661     for(x = 0; x < count; ++x)
662         pointer[x] = temp[count-x-1];
663     free(temp);
664 }
665 
666 typedef std::vector<WayPoint*> WayPointMap;
667 
668 int32 GetTimePeriodFromString(const char * str);
669 std::string ConvertTimeStampToString(uint32 timestamp);
670 std::string ConvertTimeStampToDataTime(uint32 timestamp);
671 
672 uint32 DecimalToMask(uint32 dec);
673 
674 WGSERVER_INLINE void wgs_TOLOWER(std::string& str)
675 {
676     for(size_t i = 0; i < str.length(); ++i)
677         str[i] = (char)tolower(str[i]);
678 }
679 
680 WGSERVER_INLINE void wgs_TOUPPER(std::string& str)
681 {
682     for(size_t i = 0; i < str.length(); ++i)
683         str[i] = (char)toupper(str[i]);
684 }
685 
686 // returns true if the ip hits the mask, otherwise false
687 inline static bool ParseCIDRBan(unsigned int IP, unsigned int Mask, unsigned int MaskBits)
688 {
689     // CIDR bans are a compacted form of IP / Submask
690     // So 192.168.1.0/255.255.255.0 would be 192.168.1.0/24
691     // IP's in the 192.168l.1.x range would be hit, others not.
692     unsigned char * source_ip = (unsigned char*)&IP;
693     unsigned char * mask = (unsigned char*)&Mask;
694     int full_bytes = MaskBits / 8;
695     int leftover_bits = MaskBits % 8;
696     //int byte;
697 
698     // sanity checks for the data first
699     if( MaskBits > 32 )
700         return false;
701 
702     // this is the table for comparing leftover bits
703     static const unsigned char leftover_bits_compare[9] = {
704         0x00,            // 00000000
705         0x80,            // 10000000
706         0xC0,            // 11000000
707         0xE0,            // 11100000
708         0xF0,            // 11110000
709         0xF8,            // 11111000
710         0xFC,            // 11111100
711         0xFE,            // 11111110
712         0xFF,            // 11111111 - This one isn't used
713     };
714 
715     // if we have any full bytes, compare them with memcpy
716     if( full_bytes > 0 )
717     {
718         if( memcmp( source_ip, mask, full_bytes ) != 0 )
719             return false;
720     }
721 
722     // compare the left over bits
723     if( leftover_bits > 0 )
724     {
725         if( ( source_ip[full_bytes] & leftover_bits_compare[leftover_bits] ) !=
726             ( mask[full_bytes] & leftover_bits_compare[leftover_bits] ) )
727         {
728             // one of the bits does not match
729             return false;
730         }
731     }
732 
733     // all of the bits match that were testable
734     return true;
735 }
736 
737 inline static unsigned int MakeIP(const char * str)
738 {
739     unsigned int bytes[4];
740     unsigned int res;
741     if( sscanf(str, "%u.%u.%u.%u", &bytes[0], &bytes[1], &bytes[2], &bytes[3]) != 4 )
742         return 0;
743 
744     res = bytes[0] | (bytes[1] << 8) | (bytes[2] << 16) | (bytes[3] << 24);
745     return res;
746 }
747 
748 
749 inline static unsigned int  GetANetwork(unsigned int ip)
750 {
751     return (ip & 0x000000FF);
752 }
753 
754 inline static unsigned int  GetBNetwork(unsigned int ip)
755 {
756     return (ip & 0x0000FFFF);
757 }
758 
759 inline static unsigned int  GetCNetwork(unsigned int ip)
760 {
761     return (ip & 0x00FFFFFF);
762 }
763 
764 
765 //get current thread id
766 #define getThreadId() syscall(__NR_gettid)
767 
768 // warning, by enabling this define you are aware that you are only delaying the inevitable
769 // some crashes are not recorable and those will stack up in time and lead to a full crash
770 // enabling this define will make windows servers shut down only the map instance in where the crash ocured
771 // during this forced shutdown players are not saved to avoid saving corrupted data
772 // there might be a lot of cases where each saved crash will lead to memory leaks or unhandled cases
773 // crashreports are still created and do use them to report the actaul problem that casued the crash
774 // fixing the problem that causes the crash is the proper way to fix things
775 //#define FORCED_SERVER_KEEPALIVE
776 
777 
778 #endif

另外,

在extensions/cocos-ext.h文件內加入

#include "network/TCPSocket.h"

在extensions/Android.mk文件內加入,使其在linux下可以編譯到

network/TCPSocket.cpp \

使用例子,代碼片斷:

///////////////////////////////////////////////
    // 建立SOCKET管理器
    CREATE_TCPSOCKETMGR(this);    
    // 建立並添加SOCKET,參數:服務器IP,端口,自定義的SOCKET_ID標識
    sSocketMgr.createSocket("192.168.0.183", 7502, 1);
    // 註冊協議,參數:包頭,回調函數
    sSocketMgr.register_process(0x10, SCT_CALLBACK_2(HelloWorld::login, this));
    // 註冊消息截獲事件,註冊此事件後能夠截獲收到的全部消息,若回調函數返回true則本次事件會繼續分發註冊過的協議,返回false則不分發
    sSocketMgr.register_all_process(SCT_CALLBACK_1(HelloWorld::process, this));
    // 定義協議包
    WorldPacket packet;
    packet.clear();
    // 設置協議頭
    packet.SetOpcode(0x0010);
    packet    << uint16(0x0010) 
            << uint16(0)// 協議長度
            << uint32(1) 
            << uint32(0);
    // 加入字符串數據(uint8表示字符串長度所佔字節,此處爲1字節)
    packet.AppendPacketString<uint8>(std::string("aaa:88889083:d5956683c17d7e284d33ee295b277b52"));
    // 設置協議長度
    packet.SetLength(packet.size());
    // 發送數據
    sSocketMgr.SendPacket(1, &packet);
    ///////////////////////////////////////////////

回調函數例子:

1 // 註冊單個協議回調函數(樣例),參數:SOCKET_ID標識,數據包
2     void process_login(int _tag, WorldPacket & packet);
3     // 註冊單個協議回調函數(樣例),參數:SOCKET_ID標識,協議頭,數據包
4     bool process_all(int _tag, int _cmd, WorldPacket & packet);
5     // 斷線事件
6     void onDisconnect(int _tag);

取數據例子:

WorldPacket packet ;
//
取整形數據 uint16 opcode; uint16 len; uint32 serial; float fl;
packet
>> opcode; packet >> len; packet >> serial; packet >> fl;
// 取字符串 String oldPwd, newPwd; packet.GetPacketString<uint8>(&packet, oldPwd); packet.GetPacketString<uint8>(&packet, newPwd);

寫得比較草,若是看不懂的,好的建議和意見能夠給我留言。

若是有不一樣的,好的思路也能夠提出來,一塊兒探討。。

相關文章
相關標籤/搜索