/*==LICENSE==*

CyanWorlds.com Engine - MMOG client, server and tools
Copyright (C) 2011  Cyan Worlds, Inc.

This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.

This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
GNU General Public License for more details.

You should have received a copy of the GNU General Public License
along with this program.  If not, see <http://www.gnu.org/licenses/>.

Additional permissions under GNU GPL version 3 section 7

If you modify this Program, or any covered work, by linking or
combining it with any of RAD Game Tools Bink SDK, Autodesk 3ds Max SDK,
NVIDIA PhysX SDK, Microsoft DirectX SDK, OpenSSL library, Independent
JPEG Group JPEG library, Microsoft Windows Media SDK, or Apple QuickTime SDK
(or a modified version of those libraries),
containing parts covered by the terms of the Bink SDK EULA, 3ds Max EULA,
PhysX SDK EULA, DirectX SDK EULA, OpenSSL and SSLeay licenses, IJG
JPEG Library README, Windows Media SDK EULA, or QuickTime SDK EULA, the
licensors of this Program grant you additional
permission to convey the resulting work. Corresponding Source for a
non-source form of such a combination shall include the source code for
the parts of OpenSSL and IJG JPEG Library used as well as that of the covered
work.

You can contact Cyan Worlds, Inc. by email legal@cyan.com
 or by snail mail at:
      Cyan Worlds, Inc.
      14617 N Newport Hwy
      Mead, WA   99021

*==LICENSE==*/
/*****************************************************************************
*
*   $/Plasma20/Sources/Plasma/NucleusLib/pnNetProtocol/Private/pnNpCommon.h
*   
***/

#ifdef PLASMA20_SOURCES_PLASMA_NUCLEUSLIB_PNNETPROTOCOL_PRIVATE_PNNPCOMMON_H
#error "Header $/Plasma20/Sources/Plasma/NucleusLib/pnNetProtocol/Private/pnNpCommon.h included more than once"
#endif
#define PLASMA20_SOURCES_PLASMA_NUCLEUSLIB_PNNETPROTOCOL_PRIVATE_PNNPCOMMON_H

#include "pnUUID/pnUUID.h"


/*****************************************************************************
*
*   Client message field types
*
***/

#ifdef USES_NETCLI

const NetMsgField kNetMsgFieldAccountName   = NET_MSG_FIELD_STRING(kMaxAccountNameLength);
const NetMsgField kNetMsgFieldPlayerName    = NET_MSG_FIELD_STRING(kMaxPlayerNameLength);
const NetMsgField kNetMsgFieldShaDigest     = NET_MSG_FIELD_RAW_DATA(sizeof(ShaDigest));
const NetMsgField kNetMsgFieldUuid          = NET_MSG_FIELD_DATA(sizeof(plUUID));
const NetMsgField kNetMsgFieldTransId       = NET_MSG_FIELD_DWORD();
const NetMsgField kNetMsgFieldTimeMs        = NET_MSG_FIELD_DWORD();
const NetMsgField kNetMsgFieldENetError     = NET_MSG_FIELD_DWORD();
const NetMsgField kNetMsgFieldEAgeId        = NET_MSG_FIELD_DWORD();
const NetMsgField kNetMsgFieldNetNode       = NET_MSG_FIELD_DWORD();
const NetMsgField kNetMsgFieldBuildId       = NET_MSG_FIELD_DWORD();

#endif


/*****************************************************************************
*
*   Player information structures
*
***/

#pragma pack(push,1)
struct SrvPlayerInfo {
    unsigned    playerInt;
    wchar_t       playerName[kMaxPlayerNameLength];
    wchar_t       avatarShape[kMaxVaultNodeStringLength];
    unsigned    explorer;
};
#pragma pack(pop)


/*****************************************************************************
*
*   NetAgeInfo
*
***/

struct NetAgeInfo {
    plUUID        ageInstId;
    wchar_t       ageFilename[kMaxAgeNameLength];
    wchar_t       ageInstName[kMaxAgeNameLength];
    wchar_t       ageUserName[kMaxAgeNameLength];
    wchar_t       ageDesc[1024];
    uint32_t       ageSequenceNumber;
    uint32_t       ageLanguage;
    uint32_t       population;         // only used with GetPublicAgeList query results
    uint32_t       currentPopulation;  // only used with GetPublicAgeList query results
};

/*****************************************************************************
*
*   NetGameScore
*
***/

struct NetGameScore {
    unsigned    scoreId;
    unsigned    ownerId;
    uint32_t      createdTime;
    wchar_t       gameName[kMaxGameScoreNameLength];
    unsigned    gameType;
    int         value;

    unsigned Read (const uint8_t inbuffer[], unsigned bufsz, uint8_t** end = nil);    // returns number of bytes read
    unsigned Write (ARRAY(uint8_t) * buffer) const;                                // returns number of bytes written

    void CopyFrom (const NetGameScore & score);
};

/*****************************************************************************
*
*   NetGameRank
*
***/

struct NetGameRank {
    unsigned    rank;
    int         score;
    wchar_t       name[kMaxPlayerNameLength];

    unsigned Read (const uint8_t inbuffer[], unsigned bufsz, uint8_t** end = nil);    // returns number of bytes read
    unsigned Write (ARRAY(uint8_t) * buffer) const;                                // returns number of bytes written

    void CopyFrom (const NetGameRank & fromRank);
};

/*****************************************************************************
*
*   Server vault structures
*
***/

//============================================================================
// NetVaultNode
//============================================================================
// Threaded apps: App is responsible for locking node->critsect before accessing *any* field in this struct
struct NetVaultNode : AtomicRef {
    enum RwOptions {
        kRwDirtyOnly    = 1<<0, // READ : No meaning
                                // WRITE: Only write fields marked dirty
        kRwUpdateDirty  = 1<<1, // READ : Set dirty flag on fields read from stream
                                // WRITE: Clear dirty flag on fields written to stream
    };
    
    enum CopyOptions {
        kCopySetDirty   = 1<<0,     // set dirty flag on changed dst fields
        kCopyOverwrite  = 1<<1,     // overwrite fields for which dst node already has values
        kCopyClear      = 1<<2,     // clear dst fields for which src node does not have values
    };
    
    // These flag values must not change unless all servers are
    // simultaneously replaced, so basically forget it.
    static const uint64_t kNodeId          = (uint64_t)1<< 0;
    static const uint64_t kCreateTime      = (uint64_t)1<< 1;
    static const uint64_t kModifyTime      = (uint64_t)1<< 2;
    static const uint64_t kCreateAgeName   = (uint64_t)1<< 3;
    static const uint64_t kCreateAgeUuid   = (uint64_t)1<< 4;
    static const uint64_t kCreatorAcct     = (uint64_t)1<< 5;
    static const uint64_t kCreatorId       = (uint64_t)1<< 6;
    static const uint64_t kNodeType        = (uint64_t)1<< 7;
    static const uint64_t kInt32_1         = (uint64_t)1<< 8;
    static const uint64_t kInt32_2         = (uint64_t)1<< 9;
    static const uint64_t kInt32_3         = (uint64_t)1<<10;
    static const uint64_t kInt32_4         = (uint64_t)1<<11;
    static const uint64_t kUInt32_1        = (uint64_t)1<<12;
    static const uint64_t kUInt32_2        = (uint64_t)1<<13;
    static const uint64_t kUInt32_3        = (uint64_t)1<<14;
    static const uint64_t kUInt32_4        = (uint64_t)1<<15;
    static const uint64_t kUuid_1          = (uint64_t)1<<16;
    static const uint64_t kUuid_2          = (uint64_t)1<<17;
    static const uint64_t kUuid_3          = (uint64_t)1<<18;
    static const uint64_t kUuid_4          = (uint64_t)1<<19;
    static const uint64_t kString64_1      = (uint64_t)1<<20;
    static const uint64_t kString64_2      = (uint64_t)1<<21;
    static const uint64_t kString64_3      = (uint64_t)1<<22;
    static const uint64_t kString64_4      = (uint64_t)1<<23;
    static const uint64_t kString64_5      = (uint64_t)1<<24;
    static const uint64_t kString64_6      = (uint64_t)1<<25;
    static const uint64_t kIString64_1     = (uint64_t)1<<26;
    static const uint64_t kIString64_2     = (uint64_t)1<<27;
    // blobs always come last
    static const uint64_t kText_1          = (uint64_t)1<<28;
    static const uint64_t kText_2          = (uint64_t)1<<29;
    static const uint64_t kBlob_1          = (uint64_t)1<<30;
    static const uint64_t kBlob_2          = (uint64_t)1<<31;

    static const uint64_t kAllValidFields  = 0x00000000FFFFFFFFULL;

    CCritSect   critsect;

    plUUID      revisionId;

private:
    uint64_t    fieldFlags;
    uint64_t    dirtyFlags;

    // Treat these as read-only or node flag fields will become invalid 
    // Threaded apps: Must be accessed with node->critsect locked   
    uint32_t    nodeId;
    uint32_t    createTime;
    uint32_t    modifyTime;
    wchar_t *   createAgeName;
    plUUID      createAgeUuid;
    plUUID      creatorAcct;    // accountId of node creator
    uint32_t    creatorId;      // playerId of node creator
    uint32_t    nodeType;
    int32_t     int32_1;
    int32_t     int32_2;
    int32_t     int32_3;
    int32_t     int32_4;
    uint32_t    uint32_1;
    uint32_t    uint32_2;
    uint32_t    uint32_3;
    uint32_t    uint32_4;
    plUUID      uuid_1;
    plUUID      uuid_2;
    plUUID      uuid_3;
    plUUID      uuid_4;
    wchar_t *   string64_1;
    wchar_t *   string64_2;
    wchar_t *   string64_3;
    wchar_t *   string64_4;
    wchar_t *   string64_5;
    wchar_t *   string64_6;
    wchar_t *   istring64_1;
    wchar_t *   istring64_2;
    wchar_t *   text_1;
    wchar_t *   text_2;
    uint8_t *   blob_1; size_t blob_1Length;
    uint8_t *   blob_2; size_t blob_2Length;

    void IVaultNodeSetString (
        uint64_t        bit,
        char **         pdst,
        const char      src[],
        unsigned        chars
    ) {
        free(*pdst);
        if (src && src[0])
            *pdst = StrDupLen(src, chars);
        else
            *pdst = StrDupLen("", chars);
        fieldFlags |= bit;
        dirtyFlags |= bit;
    }

    void IVaultNodeSetString (
        uint64_t        bit,
        wchar_t **      pdst,
        const wchar_t   src[],
        unsigned        chars
    ) {
        free(*pdst);
        if (src && src[0])
            *pdst = StrDupLen(src, chars);
        else
            *pdst = StrDupLen(L"", chars);
        fieldFlags |= bit;
        dirtyFlags |= bit;
    }

    template <typename T>
    inline void IVaultNodeSetValue (
        uint64_t        bit,
        T *             pdst,
        const T &       src
    ) {
        *pdst = src;
        fieldFlags |= bit;
        dirtyFlags |= bit;
    }

    void IVaultNodeSetBlob (
        uint64_t        bit,
        uint8_t **      pdst,
        unsigned *      pdstLen,
        const uint8_t   src[],
        unsigned        srcLen
    ) {
        free(*pdst);
        if (src) {
            *pdst = (uint8_t*)malloc(srcLen);
            memcpy(*pdst, src, srcLen);
            *pdstLen = srcLen;
        }
        else {
            *pdst = nil;
            *pdstLen = 0;
        }
        fieldFlags |= bit;
        dirtyFlags |= bit;
    }

    void DeallocNodeFields();

public:
    NetVaultNode ();
    ~NetVaultNode ();

    // Threaded apps: Must be called with node->critsect locked
    uint32_t Read_LCS (const uint8_t buffer[], size_t bufsz, uint32_t rwOpts);   // returns number of bytes read
    uint32_t Write_LCS (ARRAY(uint8_t) * buffer, uint32_t rwOpts);               // returns number of bytes written

    bool Matches (const NetVaultNode * other);
    void CopyFrom (const NetVaultNode * other, uint32_t copyOpts);

    uint64_t GetFieldFlags () const { return fieldFlags; }
    void ClearFieldFlags () { fieldFlags = 0; }

    uint64_t GetDirtyFlags () const { return dirtyFlags; }
    void SetDirtyFlags (uint64_t flags) { dirtyFlags |= flags; }
    void ClearDirtyFlags (uint64_t flags) { dirtyFlags &= ~flags; }

    // Threaded apps: Must be called with node->critsect locked
    void SetNodeId (uint32_t v);
    void SetNodeId_NoDirty (uint32_t v) { nodeId = v; }
    void SetCreateTime (uint32_t v);
    void SetModifyTime (uint32_t v);
    void SetCreateAgeName (const wchar_t v[]);
    void SetCreateAgeUuid (const plUUID & v);
    void SetCreatorAcct (const plUUID & v);
    void SetCreatorId (uint32_t v);
    void SetNodeType (uint32_t v);
    void SetInt32_1 (int32_t v);
    void SetInt32_2 (int32_t v);
    void SetInt32_3 (int32_t v);
    void SetInt32_4 (int32_t v);
    void SetUInt32_1 (uint32_t v);
    void SetUInt32_2 (uint32_t v);
    void SetUInt32_3 (uint32_t v);
    void SetUInt32_4 (uint32_t v);
    void SetUuid_1 (const plUUID & v);
    void SetUuid_2 (const plUUID & v);
    void SetUuid_3 (const plUUID & v);
    void SetUuid_4 (const plUUID & v);
    void SetString64_1 (const wchar_t v[]);
    void SetString64_2 (const wchar_t v[]);
    void SetString64_3 (const wchar_t v[]);
    void SetString64_4 (const wchar_t v[]);
    void SetString64_5 (const wchar_t v[]);
    void SetString64_6 (const wchar_t v[]);
    void SetIString64_1 (const wchar_t v[]);
    void SetIString64_2 (const wchar_t v[]);
    void SetText_1 (const wchar_t v[]);
    void SetText_2 (const wchar_t v[]);
    void SetBlob_1 (const uint8_t v[], size_t len);
    void SetBlob_2 (const uint8_t v[], size_t len);

    uint32_t GetNodeId () const { return nodeId; }
    uint32_t GetCreateTime () const { return createTime; }
    uint32_t GetModifyTime () const { return modifyTime; }
    const wchar_t * GetCreateAgeName () const { return createAgeName; }
    plUUID GetCreateAgeUuid () const { return createAgeUuid; }
    plUUID GetCreatorAcct () const { return creatorAcct; }
    uint32_t GetCreatorId () const { return creatorId; }
    uint32_t GetNodeType () const { return nodeType; }
    int32_t GetInt32_1 () const { return int32_1; }
    int32_t GetInt32_2 () const { return int32_2; }
    int32_t GetInt32_3 () const { return int32_3; }
    int32_t GetInt32_4 () const { return int32_4; }
    uint32_t GetUInt32_1 () const { return uint32_1; }
    uint32_t GetUInt32_2 () const { return uint32_2; }
    uint32_t GetUInt32_3 () const { return uint32_3; }
    uint32_t GetUInt32_4 () const { return uint32_4; }
    plUUID GetUuid_1 () const { return uuid_1; }
    plUUID GetUuid_2 () const { return uuid_2; }
    plUUID GetUuid_3 () const { return uuid_3; }
    plUUID GetUuid_4 () const { return uuid_4; }
    const wchar_t * GetString64_1 () const { return string64_1; }
    const wchar_t * GetString64_2 () const { return string64_2; }
    const wchar_t * GetString64_3 () const { return string64_3; }
    const wchar_t * GetString64_4 () const { return string64_4; }
    const wchar_t * GetString64_5 () const { return string64_5; }
    const wchar_t * GetString64_6 () const { return string64_6; }
    const wchar_t * GetIString64_1 () const { return istring64_1; }
    const wchar_t * GetIString64_2 () const { return istring64_2; }
    const wchar_t * GetText_1 () const { return text_1; }
    const wchar_t * GetText_2 () const { return text_2; }
    const uint8_t * GetBlob_1 () const { return blob_1; }
    size_t GetBlob_1Length () const { return blob_1Length; }
    const uint8_t * GetBlob_2 () const { return blob_2; }
    size_t GetBlob_2Length () const { return blob_2Length; }
};


//============================================================================
// NetVaultNodeRef (packed because is sent over wire directly)
//============================================================================
#pragma pack(push,1)
struct NetVaultNodeRef {
    unsigned    parentId;
    unsigned    childId;
    unsigned    ownerId;
    bool        seen;
};
#pragma pack(pop)

//============================================================================
// SrvPackBuffer
//============================================================================

// Allocate a CSrvPackBuffer on the heap with one extra uint32_t to allow for padding
#define SRV_ALLOC_BUFFER(bytes)                                     \
    new(malloc(sizeof(CSrvPackBuffer) + (bytes) + sizeof(uint32_t)))    \
    CSrvPackBuffer(bytes + sizeof(uint32_t))

// Allocate a CSrvPackBuffer on the stack with one extra uint32_t to allow for padding
#define SRV_ALLOCA_BUFFER(bytes)                                        \
    new(_alloca(sizeof(CSrvPackBuffer) + (bytes) + sizeof(uint32_t)))  \
    CSrvPackBuffer(bytes + sizeof(uint32_t))

class CSrvPackBuffer {
public:
    CSrvPackBuffer (unsigned bytes);

    void * Alloc (unsigned bytes);
    void AddData (const void * ptr, unsigned bytes);
    void AddString (const wchar_t str[]);
    void AddDWordArray (const uint32_t * arr, unsigned count);
    // add new "Add..." methods here as needed

    unsigned Size ();
    
private:
    uint8_t * m_pos;
    uint8_t * m_end;
    uint8_t * m_data;
};

class CSrvUnpackBuffer {
public:
    CSrvUnpackBuffer (const void * buffer, unsigned count);

    const void *  GetData (unsigned bytes);
    const wchar_t * GetString ();
    const uint32_t * GetDWordArray (unsigned count);
    
    unsigned BytesLeft ();
    bool ParseError ();

private:
    const uint8_t * m_pos;
    const uint8_t * m_end;
};