/*
 * PtokaX - hub server for Direct Connect peer to peer network.
 
 * Copyright (C) 2004-2017  Petr Kozelka, PPK at PtokaX dot org
 
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License version 3
 * as published by the Free Software Foundation.
 
 * 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/>.
 */
 
//---------------------------------------------------------------------------
#include "stdinc.h"
//---------------------------------------------------------------------------
#include "hashRegManager.h"
//---------------------------------------------------------------------------
#include "colUsers.h"
#include "GlobalDataQueue.h"
#include "hashUsrManager.h"
#include "LanguageManager.h"
#include "ProfileManager.h"
#include "PXBReader.h"
#include "ServerManager.h"
#include "SettingManager.h"
#include "User.h"
#include "utility.h"
//---------------------------------------------------------------------------
#ifdef _BUILD_GUI
    #include "../gui.win/RegisteredUserDialog.h"
    #include "../gui.win/RegisteredUsersDialog.h"
#endif
//---------------------------------------------------------------------------
RegManager * RegManager::m_Ptr = NULL;
//---------------------------------------------------------------------------
static const char sPtokaXRegiteredUsers[] = "PtokaX Registered Users";
static const size_t szPtokaXRegiteredUsersLen = sizeof(sPtokaXRegiteredUsers)-1;
//---------------------------------------------------------------------------
 
RegUser::RegUser() : m_tLastBadPass(0), m_sNick(NULL), m_pPrev(NULL), m_pNext(NULL), m_pHashTablePrev(NULL), m_pHashTableNext(NULL), m_ui32Hash(0), m_ui16Profile(0), m_ui8BadPassCount(0), m_bPassHash(false){
	m_sPass = NULL;
}
//---------------------------------------------------------------------------
 
RegUser::~RegUser() {
#ifdef _WIN32
    if(m_sNick != NULL && HeapFree(ServerManager::m_hPtokaXHeap, HEAP_NO_SERIALIZE, (void *)m_sNick) == 0) {
		AppendDebugLog("%s - [MEM] Cannot deallocate m_sNick in RegUser::~RegUser\n");
    }
#else
	free(m_sNick);
#endif
 
    if(m_bPassHash == true) {
#ifdef _WIN32
        if(m_ui8PassHash != NULL && HeapFree(ServerManager::m_hPtokaXHeap, HEAP_NO_SERIALIZE, (void *)m_ui8PassHash) == 0) {
		  AppendDebugLog("%s - [MEM] Cannot deallocate m_ui8PassHash in RegUser::~RegUser\n");
        }
#else
	   free(m_ui8PassHash);
#endif
    } else {
#ifdef _WIN32
        if(m_sPass != NULL && HeapFree(ServerManager::m_hPtokaXHeap, HEAP_NO_SERIALIZE, (void *)m_sPass) == 0) {
		  AppendDebugLog("%s - [MEM] Cannot deallocate m_sPass in RegUser::~RegUser\n");
        }
#else
	   free(m_sPass);
#endif
    }
}
//--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
 
RegUser * RegUser::CreateReg(char * sRegNick, const size_t szRegNickLen, char * sRegPassword, const size_t szRegPassLen, uint8_t * ui8RegPassHash, const uint16_t ui16RegProfile) {
    RegUser * pReg = new (std::nothrow) RegUser();
 
    if(pReg == NULL) {
        AppendDebugLog("%s - [MEM] Cannot allocate new Reg in RegUser::CreateReg\n");
 
        return NULL;
    }
 
#ifdef _WIN32
    pReg->m_sNick = (char *)HeapAlloc(ServerManager::m_hPtokaXHeap, HEAP_NO_SERIALIZE, szRegNickLen+1);
#else
	pReg->m_sNick = (char *)malloc(szRegNickLen+1);
#endif
    if(pReg->m_sNick == NULL) {
        AppendDebugLogFormat("[MEM] Cannot allocate %zu bytes for m_sNick in RegUser::RegUser\n", szRegNickLen+1);
 
        delete pReg;
        return NULL;
    }
    memcpy(pReg->m_sNick, sRegNick, szRegNickLen);
    pReg->m_sNick[szRegNickLen] = '\0';
 
    if(ui8RegPassHash != NULL) {
#ifdef _WIN32
        pReg->m_ui8PassHash = (uint8_t *)HeapAlloc(ServerManager::m_hPtokaXHeap, HEAP_NO_SERIALIZE, 64);
#else
        pReg->m_ui8PassHash = (uint8_t *)malloc(64);
#endif
        if(pReg->m_ui8PassHash == NULL) {
            AppendDebugLog("%s - [MEM] Cannot allocate 64 bytes for m_ui8PassHash in RegUser::RegUser\n");
 
            delete pReg;
            return NULL;
        }
        memcpy(pReg->m_ui8PassHash, ui8RegPassHash, 64);
        pReg->m_bPassHash = true;
    } else {
#ifdef _WIN32
        pReg->m_sPass = (char *)HeapAlloc(ServerManager::m_hPtokaXHeap, HEAP_NO_SERIALIZE, szRegPassLen+1);
#else
        pReg->m_sPass = (char *)malloc(szRegPassLen+1);
#endif
        if(pReg->m_sPass == NULL) {
            AppendDebugLogFormat("[MEM] Cannot allocate %zu bytes for m_sPass in RegUser::RegUser\n", szRegPassLen+1);
 
            delete pReg;
            return NULL;
        }
        memcpy(pReg->m_sPass, sRegPassword, szRegPassLen);
        pReg->m_sPass[szRegPassLen] = '\0';
    }
 
    pReg->m_ui16Profile = ui16RegProfile;
	pReg->m_ui32Hash = HashNick(sRegNick, szRegNickLen);
 
    return pReg;
}
//--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
 
bool RegUser::UpdatePassword(char * sNewPass, const size_t szNewLen) {
    if(SettingManager::m_Ptr->m_bBools[SETBOOL_HASH_PASSWORDS] == false) {
        if(m_bPassHash == true) {
            void * sOldBuf = m_ui8PassHash;
#ifdef _WIN32
			m_sPass = (char *)HeapReAlloc(ServerManager::m_hPtokaXHeap, HEAP_NO_SERIALIZE, (void *)sOldBuf, szNewLen+1);
#else
			m_sPass = (char *)realloc(sOldBuf, szNewLen+1);
#endif
            if(m_sPass == NULL) {
				m_ui8PassHash = (uint8_t *)sOldBuf;
 
                AppendDebugLogFormat("[MEM] Cannot reallocate %zu bytes for m_ui8PassHash->sPass in RegUser::UpdatePassword\n", szNewLen+1);
 
                return false;
            }
            memcpy(m_sPass, sNewPass, szNewLen);
			m_sPass[szNewLen] = '\0';
 
			m_bPassHash = false;
        } else if(strcmp(m_sPass, sNewPass) != 0) {
            char * sOldPass = m_sPass;
#ifdef _WIN32
			m_sPass = (char *)HeapReAlloc(ServerManager::m_hPtokaXHeap, HEAP_NO_SERIALIZE, (void *)sOldPass, szNewLen+1);
#else
			m_sPass = (char *)realloc(sOldPass, szNewLen+1);
#endif
            if(m_sPass == NULL) {
				m_sPass = sOldPass;
 
                AppendDebugLogFormat("[MEM] Cannot reallocate %zu bytes for m_sPass in RegUser::UpdatePassword\n", szNewLen+1);
 
                return false;
            }
            memcpy(m_sPass, sNewPass, szNewLen);
			m_sPass[szNewLen] = '\0';
        }
    } else {
        if(m_bPassHash == true) {
            HashPassword(sNewPass, szNewLen, m_ui8PassHash);
        } else {
            char * sOldPass = m_sPass;
#ifdef _WIN32
			m_ui8PassHash = (uint8_t *)HeapReAlloc(ServerManager::m_hPtokaXHeap, HEAP_NO_SERIALIZE, (void *)sOldPass, 64);
#else
			m_ui8PassHash = (uint8_t *)realloc(sOldPass, 64);
#endif
            if(m_ui8PassHash == NULL) {
				m_sPass = sOldPass;
 
                AppendDebugLog("%s - [MEM] Cannot reallocate 64 bytes for m_sPass -> m_ui8PassHash in RegUser::UpdatePassword\n");
 
                return false;
            }
 
            if(HashPassword(sNewPass, szNewLen, m_ui8PassHash) == true) {
				m_bPassHash = true;
            } else {
				m_sPass = (char *)m_ui8PassHash;
            }
        }
    }
 
    return true;
}
//--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
 
RegManager::RegManager(void) : m_ui8SaveCalls(0), m_pRegListS(NULL), m_pRegListE(NULL) {
    memset(m_pTable, 0, sizeof(m_pTable));
}
//---------------------------------------------------------------------------
 
RegManager::~RegManager(void) {
    RegUser * curReg = NULL,
        * next = m_pRegListS;
        
    while(next != NULL) {
        curReg = next;
		next = curReg->m_pNext;
 
		delete curReg;
    }
}
//---------------------------------------------------------------------------
 
bool RegManager::AddNew(char * sNick, char * sPasswd, const uint16_t iProfile) {
    if(Find(sNick, strlen(sNick)) != NULL) {
        return false;
    }
 
    RegUser * pNewUser = NULL;
 
    if(SettingManager::m_Ptr->m_bBools[SETBOOL_HASH_PASSWORDS] == true) {
        uint8_t ui8Hash[64];
 
        size_t szPassLen = strlen(sPasswd);
 
        if(HashPassword(sPasswd, szPassLen, ui8Hash) == false) {
            return false;
        }
 
        pNewUser = RegUser::CreateReg(sNick, strlen(sNick), NULL, 0, ui8Hash, iProfile);
    } else {
        pNewUser = RegUser::CreateReg(sNick, strlen(sNick), sPasswd, strlen(sPasswd), NULL, iProfile);
    }
 
    if(pNewUser == NULL) {
		AppendDebugLog("%s - [MEM] Cannot allocate pNewUser in RegManager::AddNew\n");
 
        return false;
    }
 
	Add(pNewUser);
 
#ifdef _BUILD_GUI
    if(RegisteredUsersDialog::m_Ptr != NULL) {
        RegisteredUsersDialog::m_Ptr->AddReg(pNewUser);
    }
#endif
 
    Save(true);
 
    if(ServerManager::m_bServerRunning == false) {
        return true;
    }
 
	User * AddedUser = HashManager::m_Ptr->FindUser(pNewUser->m_sNick, strlen(pNewUser->m_sNick));
 
    if(AddedUser != NULL) {
        bool bAllowedOpChat = ProfileManager::m_Ptr->IsAllowed(AddedUser, ProfileManager::ALLOWEDOPCHAT);
        AddedUser->m_i32Profile = iProfile;
 
        if(((AddedUser->m_ui32BoolBits & User::BIT_OPERATOR) == User::BIT_OPERATOR) == false) {
            if(ProfileManager::m_Ptr->IsAllowed(AddedUser, ProfileManager::HASKEYICON) == true) {
                AddedUser->m_ui32BoolBits |= User::BIT_OPERATOR;
            } else {
                AddedUser->m_ui32BoolBits &= ~User::BIT_OPERATOR;
            }
 
            if(((AddedUser->m_ui32BoolBits & User::BIT_OPERATOR) == User::BIT_OPERATOR) == true) {
				Users::m_Ptr->Add2OpList(AddedUser);
                GlobalDataQueue::m_Ptr->OpListStore(AddedUser->m_sNick);
 
                if(bAllowedOpChat != ProfileManager::m_Ptr->IsAllowed(AddedUser, ProfileManager::ALLOWEDOPCHAT)) {
					if(SettingManager::m_Ptr->m_bBools[SETBOOL_REG_OP_CHAT] == true &&
                        (SettingManager::m_Ptr->m_bBools[SETBOOL_REG_BOT] == false || SettingManager::m_Ptr->m_bBotsSameNick == false)) {
                        if(((AddedUser->m_ui32SupportBits & User::SUPPORTBIT_NOHELLO) == User::SUPPORTBIT_NOHELLO) == false) {
                            AddedUser->SendCharDelayed(SettingManager::m_Ptr->m_sPreTexts[SettingManager::SETPRETXT_OP_CHAT_HELLO],
                                SettingManager::m_Ptr->m_ui16PreTextsLens[SettingManager::SETPRETXT_OP_CHAT_HELLO]);
                        }
 
                        AddedUser->SendCharDelayed(SettingManager::m_Ptr->m_sPreTexts[SettingManager::SETPRETXT_OP_CHAT_MYINFO], SettingManager::m_Ptr->m_ui16PreTextsLens[SettingManager::SETPRETXT_OP_CHAT_MYINFO]);
                        AddedUser->SendFormat("RegManager::AddNew", true, "$OpList %s$$|", SettingManager::m_Ptr->m_sTexts[SETTXT_OP_CHAT_NICK]);
                    }
                }
            }
        }
    }
 
    return true;
}
//---------------------------------------------------------------------------
 
void RegManager::Add(RegUser * pReg) {
	Add2Table(pReg);
    
    if(m_pRegListE == NULL) {
		m_pRegListS = pReg;
		m_pRegListE = pReg;
    } else {
        pReg->m_pPrev = m_pRegListE;
		m_pRegListE->m_pNext = pReg;
		m_pRegListE = pReg;
    }
 
	return;
}
//---------------------------------------------------------------------------
 
void RegManager::Add2Table(RegUser * pReg) {
    uint16_t ui16dx = 0;
    memcpy(&ui16dx, &pReg->m_ui32Hash, sizeof(uint16_t));
 
    if(m_pTable[ui16dx] != NULL) {
		m_pTable[ui16dx]->m_pHashTablePrev = pReg;
        pReg->m_pHashTableNext = m_pTable[ui16dx];
    }
    
	m_pTable[ui16dx] = pReg;
}
//---------------------------------------------------------------------------
 
void RegManager::ChangeReg(RegUser * pReg, char * sNewPasswd, const uint16_t ui16NewProfile) {
    if(sNewPasswd != NULL) {
        size_t szPassLen = strlen(sNewPasswd);
 
        pReg->UpdatePassword(sNewPasswd, szPassLen);
    }
 
    pReg->m_ui16Profile = ui16NewProfile;
 
#ifdef _BUILD_GUI
    if(RegisteredUsersDialog::m_Ptr != NULL) {
        RegisteredUsersDialog::m_Ptr->RemoveReg(pReg);
        RegisteredUsersDialog::m_Ptr->AddReg(pReg);
    }
#endif
 
    RegManager::m_Ptr->Save(true);
 
    if(ServerManager::m_bServerRunning == false) {
        return;
    }
 
    User *ChangedUser = HashManager::m_Ptr->FindUser(pReg->m_sNick, strlen(pReg->m_sNick));
    if(ChangedUser != NULL && ChangedUser->m_i32Profile != (int32_t)ui16NewProfile) {
        bool bAllowedOpChat = ProfileManager::m_Ptr->IsAllowed(ChangedUser, ProfileManager::ALLOWEDOPCHAT);
 
        ChangedUser->m_i32Profile = (int32_t)ui16NewProfile;
 
        if(((ChangedUser->m_ui32BoolBits & User::BIT_OPERATOR) == User::BIT_OPERATOR) !=
            ProfileManager::m_Ptr->IsAllowed(ChangedUser, ProfileManager::HASKEYICON)) {
            if(ProfileManager::m_Ptr->IsAllowed(ChangedUser, ProfileManager::HASKEYICON) == true) {
                ChangedUser->m_ui32BoolBits |= User::BIT_OPERATOR;
                Users::m_Ptr->Add2OpList(ChangedUser);
                GlobalDataQueue::m_Ptr->OpListStore(ChangedUser->m_sNick);
            } else {
                ChangedUser->m_ui32BoolBits &= ~User::BIT_OPERATOR;
                Users::m_Ptr->DelFromOpList(ChangedUser->m_sNick);
            }
        }
 
        if(bAllowedOpChat != ProfileManager::m_Ptr->IsAllowed(ChangedUser, ProfileManager::ALLOWEDOPCHAT)) {
            if(ProfileManager::m_Ptr->IsAllowed(ChangedUser, ProfileManager::ALLOWEDOPCHAT) == true) {
                if(SettingManager::m_Ptr->m_bBools[SETBOOL_REG_OP_CHAT] == true &&
                    (SettingManager::m_Ptr->m_bBools[SETBOOL_REG_BOT] == false || SettingManager::m_Ptr->m_bBotsSameNick == false)) {
                    if(((ChangedUser->m_ui32SupportBits & User::SUPPORTBIT_NOHELLO) == User::SUPPORTBIT_NOHELLO) == false) {
                        ChangedUser->SendCharDelayed(SettingManager::m_Ptr->m_sPreTexts[SettingManager::SETPRETXT_OP_CHAT_HELLO],
                        SettingManager::m_Ptr->m_ui16PreTextsLens[SettingManager::SETPRETXT_OP_CHAT_HELLO]);
                    }
 
                    ChangedUser->SendCharDelayed(SettingManager::m_Ptr->m_sPreTexts[SettingManager::SETPRETXT_OP_CHAT_MYINFO], SettingManager::m_Ptr->m_ui16PreTextsLens[SettingManager::SETPRETXT_OP_CHAT_MYINFO]);
                    ChangedUser->SendFormat("RegManager::ChangeReg1", true, "$OpList %s$$|", SettingManager::m_Ptr->m_sTexts[SETTXT_OP_CHAT_NICK]);
                }
            } else {
                if(SettingManager::m_Ptr->m_bBools[SETBOOL_REG_OP_CHAT] == true && (SettingManager::m_Ptr->m_bBools[SETBOOL_REG_BOT] == false || SettingManager::m_Ptr->m_bBotsSameNick == false)) {
                    ChangedUser->SendFormat("RegManager::ChangeReg2", true, "$Quit %s|", SettingManager::m_Ptr->m_sTexts[SETTXT_OP_CHAT_NICK]);
                }
            }
        }
    }
 
#ifdef _BUILD_GUI
    if(RegisteredUserDialog::m_Ptr != NULL) {
        RegisteredUserDialog::m_Ptr->RegChanged(pReg);
    }
#endif
}
//---------------------------------------------------------------------------
 
#ifdef _BUILD_GUI
void RegManager::Delete(RegUser * pReg, const bool bFromGui/* = false*/) {
#else
void RegManager::Delete(RegUser * pReg, const bool /*bFromGui = false*/) {
#endif
	if(ServerManager::m_bServerRunning == true) {
        User * pRemovedUser = HashManager::m_Ptr->FindUser(pReg->m_sNick, strlen(pReg->m_sNick));
 
        if(pRemovedUser != NULL) {
            pRemovedUser->m_i32Profile = -1;
            if(((pRemovedUser->m_ui32BoolBits & User::BIT_OPERATOR) == User::BIT_OPERATOR) == true) {
                Users::m_Ptr->DelFromOpList(pRemovedUser->m_sNick);
                pRemovedUser->m_ui32BoolBits &= ~User::BIT_OPERATOR;
 
                if(SettingManager::m_Ptr->m_bBools[SETBOOL_REG_OP_CHAT] == true && (SettingManager::m_Ptr->m_bBools[SETBOOL_REG_BOT] == false || SettingManager::m_Ptr->m_bBotsSameNick == false)) {
                    pRemovedUser->SendFormat("RegManager::Delete", true, "$Quit %s|", SettingManager::m_Ptr->m_sTexts[SETTXT_OP_CHAT_NICK]);
                }
            }
        }
    }
 
#ifdef _BUILD_GUI
    if(bFromGui == false && RegisteredUsersDialog::m_Ptr != NULL) {
        RegisteredUsersDialog::m_Ptr->RemoveReg(pReg);
    }
#endif
 
	Rem(pReg);
 
#ifdef _BUILD_GUI
    if(RegisteredUserDialog::m_Ptr != NULL) {
        RegisteredUserDialog::m_Ptr->RegDeleted(pReg);
    }
#endif
 
    delete pReg;
 
    Save(true);
}
//---------------------------------------------------------------------------
 
void RegManager::Rem(RegUser * pReg) {
	RemFromTable(pReg);
    
    RegUser * prev, * next;
    prev = pReg->m_pPrev; next = pReg->m_pNext;
 
    if(prev == NULL) {
        if(next == NULL) {
			m_pRegListS = NULL;
			m_pRegListE = NULL;
        } else {
            next->m_pPrev = NULL;
			m_pRegListS = next;
        }
    } else if(next == NULL) {
        prev->m_pNext = NULL;
		m_pRegListE = prev;
    } else {
        prev->m_pNext = next;
        next->m_pPrev = prev;
    }
}
//---------------------------------------------------------------------------
 
void RegManager::RemFromTable(RegUser * pReg) {
    if(pReg->m_pHashTablePrev == NULL) {
        uint16_t ui16dx = 0;
        memcpy(&ui16dx, &pReg->m_ui32Hash, sizeof(uint16_t));
 
        if(pReg->m_pHashTableNext == NULL) {
			m_pTable[ui16dx] = NULL;
        } else {
            pReg->m_pHashTableNext->m_pHashTablePrev = NULL;
			m_pTable[ui16dx] = pReg->m_pHashTableNext;
        }
    } else if(pReg->m_pHashTableNext == NULL) {
        pReg->m_pHashTablePrev->m_pHashTableNext = NULL;
    } else {
        pReg->m_pHashTablePrev->m_pHashTableNext = pReg->m_pHashTableNext;
        pReg->m_pHashTableNext->m_pHashTablePrev = pReg->m_pHashTablePrev;
    }
 
	pReg->m_pHashTablePrev = NULL;
    pReg->m_pHashTableNext = NULL;
}
//---------------------------------------------------------------------------
 
RegUser* RegManager::Find(char * sNick, const size_t szNickLen) {
    uint32_t ui32Hash = HashNick(sNick, szNickLen);
 
    uint16_t ui16dx = 0;
    memcpy(&ui16dx, &ui32Hash, sizeof(uint16_t));
 
    RegUser * cur = NULL,
        * next = m_pTable[ui16dx];
 
    while(next != NULL) {
        cur = next;
        next = cur->m_pHashTableNext;
 
		if(cur->m_ui32Hash == ui32Hash && strcasecmp(cur->m_sNick, sNick) == 0) {
            return cur;
        }
    }
 
    return NULL;
}
//---------------------------------------------------------------------------
 
RegUser* RegManager::Find(User * pUser) {
    uint16_t ui16dx = 0;
    memcpy(&ui16dx, &pUser->m_ui32NickHash, sizeof(uint16_t));
 
	RegUser * cur = NULL,
        * next = m_pTable[ui16dx];
 
    while(next != NULL) {
        cur = next;
        next = cur->m_pHashTableNext;
 
		if(cur->m_ui32Hash == pUser->m_ui32NickHash && strcasecmp(cur->m_sNick, pUser->m_sNick) == 0) {
            return cur;
        }
    }
 
    return NULL;
}
//---------------------------------------------------------------------------
 
RegUser* RegManager::Find(const uint32_t ui32Hash, char * sNick) {
    uint16_t ui16dx = 0;
    memcpy(&ui16dx, &ui32Hash, sizeof(uint16_t));
 
	RegUser * cur = NULL,
        * next = m_pTable[ui16dx];
 
    while(next != NULL) {
        cur = next;
        next = cur->m_pHashTableNext;
 
		if(cur->m_ui32Hash == ui32Hash && strcasecmp(cur->m_sNick, sNick) == 0) {
            return cur;
        }
    }
 
    return NULL;
}
//---------------------------------------------------------------------------
 
void RegManager::Load(void) {
#ifdef _WIN32
    if(FileExist((ServerManager::m_sPath + "\\cfg\\RegisteredUsers.pxb").c_str()) == false) {
#else
    if(FileExist((ServerManager::m_sPath + "/cfg/RegisteredUsers.pxb").c_str()) == false) {
#endif
        LoadXML();
        return;
    }
 
    uint16_t iProfilesCount = (uint16_t)(ProfileManager::m_Ptr->m_ui16ProfileCount-1);
 
    PXBReader pxbRegs;
 
    // Open regs file
#ifdef _WIN32
    if(pxbRegs.OpenFileRead((ServerManager::m_sPath + "\\cfg\\RegisteredUsers.pxb").c_str(), 4) == false) {
#else
    if(pxbRegs.OpenFileRead((ServerManager::m_sPath + "/cfg/RegisteredUsers.pxb").c_str(), 4) == false) {
#endif
        return;
    }
 
    // Read file header
    uint16_t ui16Identificators[4] = { *((uint16_t *)"FI"), *((uint16_t *)"FV"), *((uint16_t *)"  "), *((uint16_t *)"  ") };
 
    if(pxbRegs.ReadNextItem(ui16Identificators, 2) == false) {
        return;
    }
 
    // Check header if we have correct file
    if(pxbRegs.m_ui16ItemLengths[0] != szPtokaXRegiteredUsersLen || strncmp((char *)pxbRegs.m_pItemDatas[0], sPtokaXRegiteredUsers, szPtokaXRegiteredUsersLen) != 0) {
        return;
    }
 
    {
        uint32_t ui32FileVersion = ntohl(*((uint32_t *)(pxbRegs.m_pItemDatas[1])));
 
        if(ui32FileVersion < 1) {
            return;
        }
    }
 
    // Read regs =)
    ui16Identificators[0] = *((uint16_t *)"NI");
    ui16Identificators[1] = *((uint16_t *)"PS");
    ui16Identificators[2] = *((uint16_t *)"PR");
    ui16Identificators[3] = *((uint16_t *)"PA");
 
    uint16_t iProfile = UINT16_MAX;
    RegUser * pNewUser = NULL;
    uint8_t ui8Hash[64];
    size_t szPassLen = 0;
 
    bool bSuccess = pxbRegs.ReadNextItem(ui16Identificators, 3, 1);
 
    while(bSuccess == true) {
		if(pxbRegs.m_ui16ItemLengths[0] < 65 && pxbRegs.m_ui16ItemLengths[1] < 65 && pxbRegs.m_ui16ItemLengths[2] == 2) {
            iProfile = (uint16_t)ntohs(*((uint16_t *)(pxbRegs.m_pItemDatas[2])));
 
            if(iProfile > iProfilesCount) {
                iProfile = iProfilesCount;
            }
 
            pNewUser = NULL;
 
            if(pxbRegs.m_ui16ItemLengths[3] != 0) {
                if(SettingManager::m_Ptr->m_bBools[SETBOOL_HASH_PASSWORDS] == true) {
                    szPassLen = (size_t)pxbRegs.m_ui16ItemLengths[3];
 
                    if(HashPassword((char *)pxbRegs.m_pItemDatas[3], szPassLen, ui8Hash) == false) {
                        pNewUser = RegUser::CreateReg((char *)pxbRegs.m_pItemDatas[0], pxbRegs.m_ui16ItemLengths[0], (char *)pxbRegs.m_pItemDatas[3], pxbRegs.m_ui16ItemLengths[3], NULL, iProfile);
                    } else {
                        pNewUser = RegUser::CreateReg((char *)pxbRegs.m_pItemDatas[0], pxbRegs.m_ui16ItemLengths[0], NULL, 0, ui8Hash, iProfile);
                    }
                } else {
                    pNewUser = RegUser::CreateReg((char *)pxbRegs.m_pItemDatas[0], pxbRegs.m_ui16ItemLengths[0], (char *)pxbRegs.m_pItemDatas[3], pxbRegs.m_ui16ItemLengths[3], NULL, iProfile);
                }
            } else if(pxbRegs.m_ui16ItemLengths[1] == 64) {
#ifdef _WITHOUT_SKEIN
				AppendDebugLog("%s - [ERR] Hashed password found in RegisteredUsers, but PtokaX is compiled without hashing support!\n");
 
                exit(EXIT_FAILURE);
#endif
                pNewUser = RegUser::CreateReg((char *)pxbRegs.m_pItemDatas[0], pxbRegs.m_ui16ItemLengths[0], NULL, 0, (uint8_t *)pxbRegs.m_pItemDatas[1], iProfile);
            }
 
            if(pNewUser == NULL) {
				AppendDebugLog("%s - [MEM] Cannot allocate pNewUser in RegManager::Load\n");
 
                exit(EXIT_FAILURE);
            }
 
			Add(pNewUser);
		}
 
        bSuccess = pxbRegs.ReadNextItem(ui16Identificators, 3, 1);
    }
}
//---------------------------------------------------------------------------
 
void RegManager::LoadXML() {
    uint16_t iProfilesCount = (uint16_t)(ProfileManager::m_Ptr->m_ui16ProfileCount-1);
 
#ifdef _WIN32
    TiXmlDocument doc((ServerManager::m_sPath+"\\cfg\\RegisteredUsers.xml").c_str());
#else
	TiXmlDocument doc((ServerManager::m_sPath+"/cfg/RegisteredUsers.xml").c_str());
#endif
 
    if(doc.LoadFile() == false) {
        if(doc.ErrorId() != TiXmlBase::TIXML_ERROR_OPENING_FILE && doc.ErrorId() != TiXmlBase::TIXML_ERROR_DOCUMENT_EMPTY) {
            int iMsgLen = snprintf(ServerManager::m_pGlobalBuffer, ServerManager::m_szGlobalBufferSize, "Error loading file RegisteredUsers.xml. %s (Col: %d, Row: %d)", doc.ErrorDesc(), doc.Column(), doc.Row());
			if(iMsgLen > 0) {
#ifdef _BUILD_GUI
				::MessageBox(NULL, ServerManager::m_pGlobalBuffer, g_sPtokaXTitle, MB_OK | MB_ICONERROR);
#else
				AppendLog(ServerManager::m_pGlobalBuffer);
#endif
			}
 
            exit(EXIT_FAILURE);
        }
    } else {
        TiXmlHandle cfg(&doc);
        TiXmlNode *registeredusers = cfg.FirstChild("RegisteredUsers").Node();
        if(registeredusers != NULL) {
            bool bIsBuggy = false;
            TiXmlNode *child = NULL;
 
            while((child = registeredusers->IterateChildren(child)) != NULL) {
				TiXmlNode *registereduser = child->FirstChild("Nick");
 
				if(registereduser == NULL || (registereduser = registereduser->FirstChild()) == NULL) {
					continue;
				}
 
                char *nick = (char *)registereduser->Value();
                
				if(strlen(nick) > 64 || (registereduser = child->FirstChild("Password")) == NULL ||
                    (registereduser = registereduser->FirstChild()) == NULL) {
					continue;
				}
 
                char *pass = (char *)registereduser->Value();
                
				if(strlen(pass) > 64 || (registereduser = child->FirstChild("Profile")) == NULL ||
                    (registereduser = registereduser->FirstChild()) == NULL) {
					continue;
				}
 
				uint16_t iProfile = (uint16_t)atoi(registereduser->Value());
 
				if(iProfile > iProfilesCount) {
                    int iMsgLen = snprintf(ServerManager::m_pGlobalBuffer, ServerManager::m_szGlobalBufferSize, "%s %s %s! %s %s.", LanguageManager::m_Ptr->m_sTexts[LAN_USER], nick, LanguageManager::m_Ptr->m_sTexts[LAN_HAVE_NOT_EXIST_PROFILE],
                        LanguageManager::m_Ptr->m_sTexts[LAN_CHANGED_PROFILE_TO], ProfileManager::m_Ptr->m_ppProfilesTable[iProfilesCount]->m_sName);
					if(iMsgLen > 0) {
#ifdef _BUILD_GUI
						::MessageBox(NULL, ServerManager::m_pGlobalBuffer, g_sPtokaXTitle, MB_OK | MB_ICONEXCLAMATION);
#else
						AppendLog(ServerManager::m_pGlobalBuffer);
#endif
					}
 
                    iProfile = iProfilesCount;
                    bIsBuggy = true;
                }
 
                if(Find((char*)nick, strlen(nick)) == NULL) {
                    RegUser * pNewUser = RegUser::CreateReg(nick, strlen(nick), pass, strlen(pass), NULL, iProfile);
                    if(pNewUser == NULL) {
						AppendDebugLog("%s - [MEM] Cannot allocate pNewUser in RegManager::LoadXML\n");
 
                    	exit(EXIT_FAILURE);
                    }
					Add(pNewUser);
                } else {
                    int iMsgLen = snprintf(ServerManager::m_pGlobalBuffer, ServerManager::m_szGlobalBufferSize, "%s %s %s! %s.", LanguageManager::m_Ptr->m_sTexts[LAN_USER], nick, LanguageManager::m_Ptr->m_sTexts[LAN_IS_ALREADY_IN_REGS], 
                        LanguageManager::m_Ptr->m_sTexts[LAN_USER_DELETED]);
					if(iMsgLen > 0) {
#ifdef _BUILD_GUI
						::MessageBox(NULL, ServerManager::m_pGlobalBuffer, g_sPtokaXTitle, MB_OK | MB_ICONEXCLAMATION);
#else
						AppendLog(ServerManager::m_pGlobalBuffer);
#endif
					}
 
                    bIsBuggy = true;
                }
            }
			if(bIsBuggy == true)
				Save();
        }
    }
}
//---------------------------------------------------------------------------
 
void RegManager::Save(const bool bSaveOnChange/* = false*/, const bool bSaveOnTime/* = false*/) {
    if(bSaveOnTime == true && m_ui8SaveCalls == 0) {
        return;
    }
 
	m_ui8SaveCalls++;
 
    if(bSaveOnChange == true && m_ui8SaveCalls < 100) {
        return;
    }
 
	m_ui8SaveCalls = 0;
 
    PXBReader pxbRegs;
 
    // Open regs file
#ifdef _WIN32
    if(pxbRegs.OpenFileSave((ServerManager::m_sPath + "\\cfg\\RegisteredUsers.pxb").c_str(), 3) == false) {
#else
    if(pxbRegs.OpenFileSave((ServerManager::m_sPath + "/cfg/RegisteredUsers.pxb").c_str(), 3) == false) {
#endif
        return;
    }
 
    // Write file header
    pxbRegs.m_sItemIdentifiers[0] = 'F';
    pxbRegs.m_sItemIdentifiers[1] = 'I';
    pxbRegs.m_ui16ItemLengths[0] = (uint16_t)szPtokaXRegiteredUsersLen;
    pxbRegs.m_pItemDatas[0] = (void *)sPtokaXRegiteredUsers;
    pxbRegs.m_ui8ItemValues[0] = PXBReader::PXB_STRING;
 
    pxbRegs.m_sItemIdentifiers[2] = 'F';
    pxbRegs.m_sItemIdentifiers[3] = 'V';
    pxbRegs.m_ui16ItemLengths[1] = 4;
    uint32_t ui32Version = 1;
    pxbRegs.m_pItemDatas[1] = (void *)&ui32Version;
    pxbRegs.m_ui8ItemValues[1] = PXBReader::PXB_FOUR_BYTES;
 
    if(pxbRegs.WriteNextItem(szPtokaXRegiteredUsersLen+4, 2) == false) {
        return;
    }
 
    pxbRegs.m_sItemIdentifiers[0] = 'N';
    pxbRegs.m_sItemIdentifiers[1] = 'I';
    pxbRegs.m_sItemIdentifiers[2] = 'P';
    pxbRegs.m_sItemIdentifiers[3] = 'A';
    pxbRegs.m_sItemIdentifiers[4] = 'P';
    pxbRegs.m_sItemIdentifiers[5] = 'R';
 
    pxbRegs.m_ui8ItemValues[0] = PXBReader::PXB_STRING;
    pxbRegs.m_ui8ItemValues[1] = PXBReader::PXB_STRING;
    pxbRegs.m_ui8ItemValues[2] = PXBReader::PXB_TWO_BYTES;
 
    RegUser * curReg = NULL,
        * next = m_pRegListS;
 
    while(next != NULL) {
        curReg = next;
		next = curReg->m_pNext;
 
        pxbRegs.m_ui16ItemLengths[0] = (uint16_t)strlen(curReg->m_sNick);
        pxbRegs.m_pItemDatas[0] = (void *)curReg->m_sNick;
        pxbRegs.m_ui8ItemValues[0] = PXBReader::PXB_STRING;
 
        if(curReg->m_bPassHash == true) {
            pxbRegs.m_sItemIdentifiers[3] = 'S';
 
            pxbRegs.m_ui16ItemLengths[1] = 64;
            pxbRegs.m_pItemDatas[1] = (void *)curReg->m_ui8PassHash;
        } else {
            pxbRegs.m_sItemIdentifiers[3] = 'A';
 
            pxbRegs.m_ui16ItemLengths[1] = (uint16_t)strlen(curReg->m_sPass);
            pxbRegs.m_pItemDatas[1] = (void *)curReg->m_sPass;
        }
 
        pxbRegs.m_ui16ItemLengths[2] = 2;
        pxbRegs.m_pItemDatas[2] = (void *)&curReg->m_ui16Profile;
 
        if(pxbRegs.WriteNextItem(pxbRegs.m_ui16ItemLengths[0] + pxbRegs.m_ui16ItemLengths[1] + pxbRegs.m_ui16ItemLengths[2], 3) == false) {
            break;
        }
    }
 
    pxbRegs.WriteRemaining();
}
//--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
 
void RegManager::HashPasswords() const {
    size_t szPassLen = 0;
    char * sOldPass = NULL;
 
    RegUser * pCurReg = NULL,
        * pNextReg = m_pRegListS;
 
    while(pNextReg != NULL) {
        pCurReg = pNextReg;
		pNextReg = pCurReg->m_pNext;
 
        if(pCurReg->m_bPassHash == false) {
            sOldPass = pCurReg->m_sPass;
#ifdef _WIN32
            pCurReg->m_ui8PassHash = (uint8_t *)HeapAlloc(ServerManager::m_hPtokaXHeap, HEAP_NO_SERIALIZE, 64);
#else
            pCurReg->m_ui8PassHash = (uint8_t *)malloc(64);
#endif
            if(pCurReg->m_ui8PassHash == NULL) {
                pCurReg->m_sPass = sOldPass;
 
                AppendDebugLog("%s - [MEM] Cannot reallocate 64 bytes for sPass->ui8PassHash in RegManager::HashPasswords\n");
 
                continue;
            }
 
            szPassLen = strlen(sOldPass);
 
            if(HashPassword(sOldPass, szPassLen, pCurReg->m_ui8PassHash) == true) {
                pCurReg->m_bPassHash = true;
 
#ifdef _WIN32
                if(HeapFree(ServerManager::m_hPtokaXHeap, HEAP_NO_SERIALIZE, (void *)sOldPass) == 0) {
                    AppendDebugLog("%s - [MEM] Cannot deallocate sOldPass in RegManager::HashPasswords\n");
                }
#else
                free(sOldPass);
#endif
            } else {
#ifdef _WIN32
                if(HeapFree(ServerManager::m_hPtokaXHeap, HEAP_NO_SERIALIZE, (void *)pCurReg->m_ui8PassHash) == 0) {
                    AppendDebugLog("%s - [MEM] Cannot deallocate pCurReg->ui8PassHash in RegManager::HashPasswords\n");
                }
#else
                free(pCurReg->m_ui8PassHash);
#endif
 
                pCurReg->m_sPass = sOldPass;
            }
        }
    }
}
//--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
 
void RegManager::AddRegCmdLine() {
	char sNick[66];
 
nick:
	printf("Please enter Nick for new Registered User (Maximal length 64 characters. Characters |, $ and space are not allowed): ");
	if(fgets(sNick, 66, stdin) != NULL) {
		char * sMatch = strchr(sNick, '\n');
		if(sMatch != NULL) {
			sMatch[0] = '\0';
		}
 
		sMatch = strpbrk(sNick, " $|");
		if(sMatch != NULL) {
			printf("Character '%c' is not allowed in Nick!\n", sMatch[0]);
 
			if(WantAgain() == false) {
				return;
			}
 
			goto nick;
		}
 
		size_t szLen = strlen(sNick);
 
		if(szLen == 0) {
			printf("No Nick specified!\n");
 
			if(WantAgain() == false) {
				return;
			}
 
			goto nick;
		}
 
		RegUser * pReg = Find(sNick, strlen(sNick));
		if(pReg != NULL) {
			printf("Registered user with nick '%s' already exist!\n", sNick);
 
			if(WantAgain() == false) {
				return;
			}
 
			goto nick;
		}
	} else {
		printf("Error reading Nick... ending.\n");
		exit(EXIT_FAILURE);
	}
 
	char sPassword[66];
 
password:
	printf("Please enter Password for new Registered User (Maximal length 64 characters. Character | is not allowed): ");
	if(fgets(sPassword, 66, stdin) != NULL) {
		char * sMatch = strchr(sPassword, '\n');
		if(sMatch != NULL) {
			sMatch[0] = '\0';
		}
 
		sMatch = strchr(sPassword, '|');
		if(sMatch != NULL) {
			printf("Character | is not allowed in Password!\n");
 
			if(WantAgain() == false) {
				return;
			}
 
			goto password;
		}
 
		size_t szLen = strlen(sPassword);
 
		if(szLen == 0) {
			printf("No Password specified!\n");
 
			if(WantAgain() == false) {
				return;
			}
 
			goto password;
		}
	} else {
		printf("Error reading Password... ending.\n");
		exit(EXIT_FAILURE);
	}
 
	printf("\nAvailable profiles: \n");
    for(uint16_t ui16i = 0; ui16i < ProfileManager::m_Ptr->m_ui16ProfileCount; ui16i++) {
    	printf("%hu - %s\n", ui16i, ProfileManager::m_Ptr->m_ppProfilesTable[ui16i]->m_sName);
    }
 
	uint16_t ui16Profile = 0;
	char sProfile[7];
 
profile:
 
	printf("Please enter Profile number for new Registered User: ");
	if(fgets(sProfile, 7, stdin) != NULL) {
		char * sMatch = strchr(sProfile, '\n');
		if(sMatch != NULL) {
			sMatch[0] = '\0';
		}
 
		uint8_t ui8Len = (uint8_t)strlen(sProfile);
 
		if(ui8Len == 0) {
			printf("No Profile specified!\n");
 
			if(WantAgain() == false) {
				return;
			}
 
			goto profile;
		}
 
		for(uint8_t ui8i = 0; ui8i < ui8Len; ui8i++) {
			if(isdigit(sProfile[ui8i]) == 0) {
				printf("Character '%c' is not valid number!\n", sProfile[ui8i]);
 
				if(WantAgain() == false) {
					return;
				}
 
				goto profile;
			}
		}
 
		ui16Profile = (uint16_t)atoi(sProfile);
		if(ui16Profile >= ProfileManager::m_Ptr->m_ui16ProfileCount) {
			printf("Profile number %hu not exist!\n", ui16Profile);
 
			if(WantAgain() == false) {
				return;
			}
 
			goto profile;
		}
	} else {
		printf("Error reading Profile... ending.\n");
		exit(EXIT_FAILURE);
	}
 
	if(AddNew(sNick, sPassword, ui16Profile) == false) {
		printf("Error adding new Registered User... ending.\n");
		exit(EXIT_FAILURE);
	} else {
		printf("Registered User with Nick '%s' Password '%s' and Profile '%hu' was added.", sNick, sPassword, ui16Profile);
	}
}
//--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------

V512 A call of the 'memcpy' function will lead to underflow of the buffer '& ui32Hash'.

V512 A call of the 'memcpy' function will lead to underflow of the buffer '& pReg->m_ui32Hash'.

V512 A call of the 'memcpy' function will lead to underflow of the buffer '& ui32Hash'.

V512 A call of the 'memcpy' function will lead to underflow of the buffer '& pReg->m_ui32Hash'.

V512 A call of the 'memcpy' function will lead to underflow of the buffer '& pUser->m_ui32NickHash'.

V525 The code contains the collection of similar blocks. Check items ''N'', ''I'', ''P'', ''A'', ''P'', ''R'' in lines 789, 790, 791, 792, 793, 794.