331 lines
14 KiB
331 lines
14 KiB
/*==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==*/ |
|
|
|
#include "hsWindows.h" |
|
#include "hsTimer.h" |
|
#include "pfDispatchLog.h" |
|
#include "plStatusLog/plStatusLog.h" |
|
#include "pnMessage/plMessage.h" |
|
#include "pnKeyedObject/plKey.h" |
|
#include "plString.h" |
|
|
|
static bool DumpSpecificMsgInfo(plMessage* msg, plString& info); |
|
|
|
plDispatchLog::plDispatchLog() : |
|
fLog(nil), |
|
fStartTicks(hsTimer::GetTicks()) |
|
{ |
|
fLog = plStatusLogMgr::GetInstance().CreateStatusLog(20, "Dispatch.log", plStatusLog::kAlignToTop | plStatusLog::kFilledBackground | plStatusLog::kRawTimeStamp); |
|
fIncludeTypes.SetSize(plFactory::GetNumClasses()); |
|
} |
|
|
|
plDispatchLog::~plDispatchLog() |
|
{ |
|
delete fLog; |
|
} |
|
|
|
void plDispatchLog::InitInstance() |
|
{ |
|
static plDispatchLog dispatchLog; |
|
fInstance = &dispatchLog; |
|
} |
|
|
|
void plDispatchLog::LogStatusBarChange(const char* name, const char* action) |
|
{ |
|
fLog->AddLineF("----- Status bar '%s' %s -----", name, action); |
|
|
|
#ifdef HS_BUILD_FOR_WIN32 |
|
MEMORYSTATUS ms; |
|
GlobalMemoryStatus(&ms); |
|
|
|
MEMORY_BASIC_INFORMATION mbi; |
|
memset(&mbi, 0, sizeof(MEMORY_BASIC_INFORMATION)); |
|
|
|
// Note: this will return shared mem too on Win9x. There's a way to catch that, but it's too slow -Colin |
|
uint32_t processMemUsed = 0; |
|
void* curAddress = 0; |
|
while (VirtualQuery(curAddress, &mbi, sizeof(MEMORY_BASIC_INFORMATION)) == sizeof(MEMORY_BASIC_INFORMATION)) |
|
{ |
|
if (mbi.State == MEM_COMMIT && mbi.Type == MEM_PRIVATE) |
|
processMemUsed += mbi.RegionSize; |
|
curAddress = ((BYTE*)mbi.BaseAddress) + mbi.RegionSize; |
|
} |
|
|
|
#define ToMB(mem) float(mem) / (1024.f*1024.f) |
|
fLog->AddLineF("# Mem stats"); |
|
fLog->AddLineF("# Physical: %.1f MB used %.1f MB free", ToMB(ms.dwTotalPhys-ms.dwAvailPhys), ToMB(ms.dwAvailPhys)); |
|
fLog->AddLineF("# Virtual: %.1f MB used %.1f MB free", ToMB(ms.dwTotalVirtual-ms.dwAvailVirtual), ToMB(ms.dwAvailVirtual)); |
|
fLog->AddLineF("# Pagefile: %.1f MB used %.1f MB free", ToMB(ms.dwTotalPageFile-ms.dwAvailPageFile), ToMB(ms.dwAvailPageFile)); |
|
fLog->AddLineF("# Process: %.1f MB used", ToMB(processMemUsed)); |
|
#endif // HS_BUILD_FOR_WIN32 |
|
} |
|
|
|
void plDispatchLog::LogLongReceive(const char* keyname, const char* className, uint32_t clonePlayerID, plMessage* msg, float ms) |
|
{ |
|
plString info; |
|
if (DumpSpecificMsgInfo(msg, info)) |
|
fLog->AddLineF("%-30s[%7u](%-20s) took %6.1f ms to receive %s[%s]\n", keyname, clonePlayerID, className, ms, msg->ClassName(), info.c_str()); |
|
else |
|
fLog->AddLineF("%-30s[%7u](%-20s) took %6.1f ms to receive %s\n", keyname, clonePlayerID, className, ms, msg->ClassName()); |
|
} |
|
|
|
void plDispatchLog::DumpMsg(plMessage* msg, int numReceivers, int sendTimeMs, int32_t indent) |
|
{ |
|
if (!msg) |
|
return; |
|
|
|
bool found=fIncludeTypes.IsBitSet(msg->ClassIndex()); |
|
if (found && !hsCheckBits(fFlags, plDispatchLogBase::kInclude)) |
|
// it's an exclude list and we found it |
|
return; |
|
if (!found && hsCheckBits(fFlags, plDispatchLogBase::kInclude)) |
|
// it's an include list and we didn't find it |
|
return; |
|
|
|
static float lastTime=0; |
|
float curTime = (float)hsTimer::GetSysSeconds(); |
|
|
|
if (lastTime!=curTime) |
|
{ |
|
// add linebreak for new frame |
|
fLog->AddLine("\n"); |
|
} |
|
|
|
float sendTime = hsTimer::GetMilliSeconds<float>(hsTimer::GetTicks() - fStartTicks); |
|
|
|
char indentStr[50]; |
|
indent = std::min(indent, static_cast<int32_t>(sizeof(indentStr)-1)); |
|
memset(indentStr, ' ', indent); |
|
indentStr[indent] = '\0'; |
|
|
|
fLog->AddLineF("%sDispatched (%d) %d ms: time=%d CName=%s, sndr=%s, rcvr(%d)=%s, flags=0x%lx, tstamp=%f\n", |
|
indentStr, numReceivers, sendTimeMs, |
|
int(sendTime), msg->ClassName(), msg->fSender?msg->fSender->GetName().c_str():"nil", |
|
msg->GetNumReceivers(), msg->GetNumReceivers() && msg->GetReceiver(0) |
|
? msg->GetReceiver(0)->GetName().c_str():"nil", |
|
msg->fBCastFlags, msg->fTimeStamp); |
|
|
|
lastTime=curTime; |
|
} |
|
|
|
void plDispatchLog::AddFilterType(uint16_t hClass) |
|
{ |
|
if (hClass>=plFactory::GetNumClasses()) |
|
return; |
|
|
|
int i; |
|
for( i = 0; i < plFactory::GetNumClasses(); i++ ) |
|
{ |
|
if( plFactory::DerivesFrom(hClass, i) ) |
|
AddFilterExactType(i); |
|
} |
|
} |
|
|
|
void plDispatchLog::AddFilterExactType(uint16_t type) |
|
{ |
|
if (type<plFactory::GetNumClasses()) |
|
fIncludeTypes.SetBit(type); |
|
} |
|
|
|
void plDispatchLog::RemoveFilterType(uint16_t hClass) |
|
{ |
|
if (hClass>=plFactory::GetNumClasses()) |
|
return; |
|
|
|
int i; |
|
for( i = 0; i < plFactory::GetNumClasses(); i++ ) |
|
{ |
|
if( plFactory::DerivesFrom(hClass, i) ) |
|
RemoveFilterExactType(i); |
|
} |
|
} |
|
|
|
void plDispatchLog::RemoveFilterExactType(uint16_t type) |
|
{ |
|
if (type<plFactory::GetNumClasses()) |
|
fIncludeTypes.ClearBit(type); |
|
} |
|
|
|
|
|
////////////////////////////////////////////////////////////////////////// |
|
////////////////////////////////////////////////////////////////////////// |
|
|
|
#include "pnMessage/plClientMsg.h" |
|
#include "pfMessage/pfKIMsg.h" |
|
#include "pnKeyedObject/hsKeyedObject.h" |
|
#include "plResMgr/plKeyFinder.h" |
|
#include "plResMgr/plPageInfo.h" |
|
|
|
static bool DumpSpecificMsgInfo(plMessage* msg, plString& info) |
|
{ |
|
#ifndef PLASMA_EXTERNAL_RELEASE // Don't bloat up the external release with all these strings |
|
pfKIMsg* kiMsg = pfKIMsg::ConvertNoRef(msg); |
|
if (kiMsg) |
|
{ |
|
const char* typeName = "(unknown)"; |
|
#define PrintKIType(type) if (kiMsg->GetCommand() == pfKIMsg::type) typeName = #type; |
|
PrintKIType(kHACKChatMsg); // send chat message via pfKIMsg |
|
PrintKIType(kEnterChatMode); // toggle chat mode |
|
PrintKIType(kSetChatFadeDelay); // set the chat delay |
|
PrintKIType(kSetTextChatAdminMode); // set the chat admin mode... not used (see CCR) |
|
PrintKIType(kDisableKIandBB); // disable KI and blackbar (for things like AvaCusta) |
|
PrintKIType(kEnableKIandBB); // re-enable the KI and blackbar |
|
PrintKIType(kYesNoDialog); // display a Yes/No dialog |
|
PrintKIType(kAddPlayerDevice); // add a device player list); such as imager |
|
PrintKIType(kRemovePlayerDevice); // remove a device from player list |
|
PrintKIType(kUpgradeKILevel); // upgrade the KI to higher level |
|
PrintKIType(kDowngradeKILevel); // downgrade KI to next lower level |
|
PrintKIType(kRateIt); // display the "RateIt"(tm) dialog |
|
PrintKIType(kSetPrivateChatChannel); // set the private chat channel (for private rooms) |
|
PrintKIType(kUnsetPrivateChatChannel); // unset private chat channel |
|
PrintKIType(kStartBookAlert); // blink the book image on the blackbar |
|
PrintKIType(kMiniBigKIToggle); // shortcut to toggling the miniKI/bigKI |
|
PrintKIType(kKIPutAway); // shortcut to hiding all of the KI |
|
PrintKIType(kChatAreaPageUp); // shortcut to paging up the chat area |
|
PrintKIType(kChatAreaPageDown); // shortcut to paging down the chat area |
|
PrintKIType(kChatAreaGoToBegin); // shortcut to going to the beginning of the chat area |
|
PrintKIType(kChatAreaGoToEnd); // shortcut to going to the end of the chat area |
|
PrintKIType(kKITakePicture); // shortcut to taking a picture in the KI |
|
PrintKIType(kKICreateJournalNote); // shortcut to creating a journal note in the KI |
|
PrintKIType(kKIToggleFade); // shortcut to toggle fade mode |
|
PrintKIType(kKIToggleFadeEnable); // shortcut to toggling fade enabled |
|
PrintKIType(kKIChatStatusMsg); // display status message in chat window |
|
PrintKIType(kKILocalChatStatusMsg); // display status message in chat window |
|
PrintKIType(kKIUpSizeFont); // bump up the size of the chat area font |
|
PrintKIType(kKIDownSizeFont); // down size the font of the chat area |
|
PrintKIType(kKIOpenYeehsaBook); // open the playerbook if not already open |
|
PrintKIType(kKIOpenKI); // open up in degrees the KI |
|
PrintKIType(kKIShowCCRHelp); // show the CCR help dialog |
|
PrintKIType(kKICreateMarker); // create a marker |
|
PrintKIType(kKICreateMarkerFolder); // create a marker folder in the current Age's journal folder |
|
PrintKIType(kKILocalChatErrorMsg); // display error message in chat window |
|
PrintKIType(kKIPhasedAllOn); // turn on all the phased KI functionality |
|
PrintKIType(kKIPhasedAllOff); // turn off all the phased KI functionality |
|
PrintKIType(kKIOKDialog); // display an OK dialog box (localized) |
|
PrintKIType(kDisableYeeshaBook); // don't allow linking with the Yeesha book (gameplay) |
|
PrintKIType(kEnableYeeshaBook); // re-allow linking with the Yeesha book |
|
PrintKIType(kQuitDialog); // put up quit dialog |
|
PrintKIType(kTempDisableKIandBB); // temp disable KI and blackbar (done by av system) |
|
PrintKIType(kTempEnableKIandBB); // temp re-enable the KI and blackbar (done by av system) |
|
PrintKIType(kDisableEntireYeeshaBook); // disable the entire Yeeshabook); not for gameplay); but prevent linking |
|
PrintKIType(kEnableEntireYeeshaBook); |
|
PrintKIType(kKIOKDialogNoQuit); // display OK dialog in the KI without quiting afterwards |
|
PrintKIType(kGZUpdated); // the GZ game was updated |
|
PrintKIType(kGZInRange); // a GZ marker is in range |
|
PrintKIType(kGZOutRange); // GZ markers are out of range |
|
PrintKIType(kUpgradeKIMarkerLevel); // upgrade the KI Marker level (current 0 and 1) |
|
PrintKIType(kKIShowMiniKI); // force the miniKI up |
|
PrintKIType(kGZFlashUpdate); // flash an update without saving (for animation of GZFill in) |
|
PrintKIType(kNoCommand); |
|
|
|
info = plFormat("Type: {} Str: {} User: {}({}) Delay: {} Int: {}", |
|
typeName, |
|
kiMsg->GetString(), |
|
kiMsg->GetUser(), |
|
kiMsg->GetPlayerID(), |
|
kiMsg->GetDelay(), |
|
kiMsg->GetIntValue()); |
|
|
|
return true; |
|
} |
|
|
|
plClientMsg* clientMsg = plClientMsg::ConvertNoRef(msg); |
|
if (clientMsg) |
|
{ |
|
#define PrintType(type) if (clientMsg->GetClientMsgFlag() == plClientMsg::type) info = #type; |
|
PrintType(kLoadRoom); |
|
PrintType(kLoadRoomHold); |
|
PrintType(kUnloadRoom); |
|
PrintType(kLoadNextRoom); |
|
PrintType(kInitComplete); |
|
PrintType(kDisableRenderScene); |
|
PrintType(kEnableRenderScene); |
|
PrintType(kQuit); |
|
PrintType(kLoadAgeKeys); |
|
PrintType(kReleaseAgeKeys); |
|
|
|
switch (clientMsg->GetClientMsgFlag()) |
|
{ |
|
case plClientMsg::kLoadRoom: |
|
case plClientMsg::kLoadRoomHold: |
|
case plClientMsg::kUnloadRoom: |
|
{ |
|
info += " - Pages: "; |
|
|
|
const std::vector<plLocation>& locs = clientMsg->GetRoomLocs(); |
|
for (int i = 0; i < locs.size(); i++) |
|
{ |
|
const plLocation& loc = locs[i]; |
|
const plPageInfo* pageInfo = plKeyFinder::Instance().GetLocationInfo(loc); |
|
|
|
if (pageInfo) |
|
info += plFormat("{}-{} ", pageInfo->GetAge(), pageInfo->GetPage()); |
|
} |
|
} |
|
break; |
|
|
|
case plClientMsg::kLoadAgeKeys: |
|
case plClientMsg::kReleaseAgeKeys: |
|
info += plFormat(" - Age: {}", clientMsg->GetAgeName()); |
|
break; |
|
} |
|
return true; |
|
} |
|
|
|
plRefMsg* refMsg = plRefMsg::ConvertNoRef(msg); |
|
if (refMsg) |
|
{ |
|
const char* typeName = nil; |
|
#define GetType(type) if (refMsg->GetContext() == plRefMsg::type) typeName = #type; |
|
GetType(kOnCreate); |
|
GetType(kOnDestroy); |
|
GetType(kOnRequest); |
|
GetType(kOnRemove); |
|
GetType(kOnReplace); |
|
info = plFormat("Obj: {} RefType: {}", refMsg->GetRef()->GetKeyName(), typeName); |
|
|
|
return true; |
|
} |
|
#endif // PLASMA_EXTERNAL_RELEASE |
|
|
|
return false; |
|
}
|
|
|