/*==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 . 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 "hsTimer.h" #include "hsResMgr.h" #include "plNetClientMgr.h" #include "plCreatableIndex.h" #include "plNetObjectDebugger.h" #include "plNetClientMsgScreener.h" #include "../pnNetCommon/plSynchedObject.h" #include "../pnNetCommon/plSDLTypes.h" #include "../pnMessage/plCameraMsg.h" #include "../plNetClientRecorder/plNetClientRecorder.h" #include "../plMessage/plLoadCloneMsg.h" #include "../plMessage/plLoadAvatarMsg.h" #include "../plAvatar/plAvatarClothing.h" #include "../plAvatar/plArmatureMod.h" #include "../plAvatar/plAvatarMgr.h" #include "../plNetMessage/plNetMessage.h" #include "../plMessage/plCCRMsg.h" #include "../plVault/plVault.h" #include "../plContainer/plConfigInfo.h" #include "../plDrawable/plMorphSequence.h" #include "../plParticleSystem/plParticleSystem.h" #include "../plParticleSystem/plParticleSDLMod.h" #include "../plResMgr/plLocalization.h" #include "../../FeatureLib/pfMessage/pfKIMsg.h" // TMP #include "../plNetGameLib/plNetGameLib.h" #include "../plSDL/plSDL.h" // // request members list from server // int plNetClientMgr::ISendMembersListRequest() { plNetMsgMembersListReq msg; msg.SetNetProtocol(kNetProtocolCli2Game); return SendMsg(&msg); } // // reset paged in rooms list on server // int plNetClientMgr::ISendRoomsReset() { plNetMsgPagingRoom msg; msg.SetPageFlags(plNetMsgPagingRoom::kResetList); msg.SetNetProtocol(kNetProtocolCli2Game); return SendMsg(&msg); } // // Make sure all dirty objects save their state. // Mark those objects as clean and clear the dirty list. // int plNetClientMgr::ISendDirtyState(double secs) { std::vector carryOvers; Int32 num=plSynchedObject::GetNumDirtyStates(); #if 0 if (num) { DebugMsg("%d dirty sdl state msgs queued, t=%f", num, secs); } #endif Int32 i; for(i=0;iGetObject(); if (!obj) continue; // could add to carryOvers if (!(state->fSendFlags & plSynchedObject::kSkipLocalOwnershipCheck)) { int localOwned=obj->IsLocallyOwned(); if (localOwned==plSynchedObject::kNo) { DebugMsg("Late rejection of queued SDL state, obj %s, sdl %s", state->fObjKey->GetName(), state->fSDLName.c_str()); continue; } } obj->CallDirtyNotifiers(); obj->SendSDLStateMsg(state->fSDLName.c_str(), state->fSendFlags); } plSynchedObject::ClearDirtyState(carryOvers); return hsOK; } // // Given a plasma petition msg, send a petition text node to the vault // vault will detect and fwd to CCR system. // void plNetClientMgr::ISendCCRPetition(plCCRPetitionMsg* petMsg) { // petition msg info UInt8 type = petMsg->GetType(); const char* title = petMsg->GetTitle(); const char* note = petMsg->GetNote(); std::string work = note; std::replace( work.begin(), work.end(), '\n', '\t' ); note = work.c_str(); // stuff petition info fields into a config info object plConfigInfo info; info.AddValue( "Petition", "Type", type ); info.AddValue( "Petition", "Content", note ); info.AddValue( "Petition", "Title", title ); info.AddValue( "Petition", "Language", plLocalization::GetLanguageName( plLocalization::GetLanguage() ) ); info.AddValue( "Petition", "AcctName", NetCommGetAccount()->accountNameAnsi ); char buffy[20]; sprintf( buffy, "%lu", GetPlayerID() ); info.AddValue( "Petition", "PlayerID", buffy ); info.AddValue( "Petition", "PlayerName", GetPlayerName() ); // write config info formatted like an ini file to a buffer hsRAMStream ram; info.WriteTo( &plIniStreamConfigSource( &ram ) ); int size = ram.GetPosition(); ram.Rewind(); std::string buf; buf.resize( size ); ram.CopyToMem( (void*)buf.data() ); wchar * wStr = StrDupToUnicode(buf.c_str()); NetCliAuthSendCCRPetition(wStr); FREE(wStr); } // // send a msg to reset the camera in a new age // void plNetClientMgr::ISendCameraReset(hsBool bEnteringAge) { plCameraMsg* pCamMsg = TRACKED_NEW plCameraMsg; if (bEnteringAge) pCamMsg->SetCmd(plCameraMsg::kResetOnEnter); else pCamMsg->SetCmd(plCameraMsg::kResetOnExit); pCamMsg->SetBCastFlag(plMessage::kBCastByExactType, false); plUoid U2(kVirtualCamera1_KEY); plKey pCamKey = hsgResMgr::ResMgr()->FindKey(U2); if (pCamKey) pCamMsg->AddReceiver(pCamKey); pCamMsg->Send(); } // // When we link in to a new age, we need to send our avatar state up to the gameserver // void plNetClientMgr::SendLocalPlayerAvatarCustomizations() { plSynchEnabler ps(true); // make sure synching is enabled, since this happens during load const plArmatureMod * avMod = plAvatarMgr::GetInstance()->GetLocalAvatar(); hsAssert(avMod,"Failed to get local avatar armature modifier."); avMod->GetClothingOutfit()->DirtySynchState(kSDLClothing, plSynchedObject::kBCastToClients | plSynchedObject::kForceFullSend); plSceneObject* pObj = (const_cast(avMod))->GetFollowerParticleSystemSO(); if (pObj) { const plParticleSystem* sys = plParticleSystem::ConvertNoRef(pObj->GetModifierByType(plParticleSystem::Index())); if (sys) (const_cast(sys))->GetSDLMod()->SendState(plSynchedObject::kBCastToClients | plSynchedObject::kForceFullSend); } // may want to do this all the time, but for now stealthmode is the only extra avatar state we care about // don't bcast this to other clients, the invis level is contained in the linkIn msg which will synch other clients if (avMod->IsInStealthMode() && avMod->GetTarget(0)) avMod->GetTarget(0)->DirtySynchState(kSDLAvatar, plSynchedObject::kForceFullSend); hsTArray morphs; plMorphSequence::FindMorphMods(avMod->GetTarget(0), morphs); int i; for (i = 0; i < morphs.GetCount(); i++) if (morphs[i]->GetTarget(0)) morphs[i]->GetTarget(0)->DirtySynchState(kSDLMorphSequence, plSynchedObject::kBCastToClients); } // // Called to send a plasma msg out over the network. Called by the dispatcher. // return hsOK if ok // int plNetClientMgr::ISendGameMessage(plMessage* msg) { if (GetFlagsBit(kDisabled)) return hsOK; static plNetClientMsgScreener screener; // make static so that there's only 1 log per session if (!screener.AllowMessage(msg)) { if (GetFlagsBit(kScreenMessages)) return hsOK; // filter out illegal messages } // TEMP if (GetFlagsBit(kSilencePlayer)) { pfKIMsg* kiMsg = pfKIMsg::ConvertNoRef(msg); if (kiMsg && kiMsg->GetCommand()==pfKIMsg::kHACKChatMsg) return hsOK; } plNetPlayerIDList* dstIDs = msg->GetNetReceivers(); #ifdef HS_DEBUGGING if ( dstIDs ) { DebugMsg( "Preparing to send %s to specific players.", msg->ClassName() ); } #endif // get sender object plSynchedObject* synchedObj = msg->GetSender() ? plSynchedObject::ConvertNoRef(msg->GetSender()->ObjectIsLoaded()) : nil; // if sender is flagged as localOnly, he shouldn't talk to the network if (synchedObj && !synchedObj->IsNetSynched() ) return hsOK; // choose appropriate type of net game msg wrapper plNetMsgGameMessage* netMsgWrap=nil; plLoadCloneMsg* loadClone = plLoadCloneMsg::ConvertNoRef(msg); if (loadClone) { plLoadAvatarMsg* lam=plLoadAvatarMsg::ConvertNoRef(msg); netMsgWrap = TRACKED_NEW plNetMsgLoadClone; plNetMsgLoadClone* netLoadClone=plNetMsgLoadClone::ConvertNoRef(netMsgWrap); netLoadClone->SetIsPlayer(lam && lam->GetIsPlayer()); netLoadClone->SetIsLoading(loadClone->GetIsLoading()!=0); netLoadClone->ObjectInfo()->SetFromKey(loadClone->GetCloneKey()); } else if (dstIDs) { netMsgWrap = TRACKED_NEW plNetMsgGameMessageDirected; int i; for(i=0;isize();i++) { UInt32 playerID = (*dstIDs)[i]; if (playerID == NetCommGetPlayer()->playerInt) continue; hsLogEntry( DebugMsg( "\tAdding receiver: %lu" , playerID ) ); ((plNetMsgGameMessageDirected*)netMsgWrap)->Receivers()->AddReceiverPlayerID( playerID ); } } else netMsgWrap = TRACKED_NEW plNetMsgGameMessage; // check delivery timestamp if (msg->GetTimeStamp()<=hsTimer::GetSysSeconds()) msg->SetTimeStamp(0); else netMsgWrap->GetDeliveryTime().SetFromGameTime(msg->GetTimeStamp(), hsTimer::GetSysSeconds()); // write message (and label) to ram stream hsRAMStream stream; hsgResMgr::ResMgr()->WriteCreatable(&stream, msg); // put stream in net msg wrapper netMsgWrap->StreamInfo()->CopyStream(&stream); // hsLogEntry( DebugMsg(plDispatchLog::GetInstance()->MakeMsgInfoString(msg, "\tActionMsg:",0)) ); // check if this msg uses direct communication (sent to specific rcvrs) // if so the server can filter it hsBool bCast = msg->HasBCastFlag(plMessage::kBCastByExactType) || msg->HasBCastFlag(plMessage::kBCastByType); hsBool directCom = msg->GetNumReceivers()>0; if( directCom ) { // It's direct if we have receivers AND any of them are in non-virtual locations int i; for( i = 0, directCom = false; i < msg->GetNumReceivers(); i++ ) { if( !msg->GetReceiver( i )->GetUoid().GetLocation().IsVirtual() && !msg->GetReceiver( i )->GetUoid().GetLocation().IsReserved() // && !IsBuiltIn ) { directCom = true; break; } } if (!directCom) bCast = true; } if (!directCom && !bCast && !dstIDs) WarningMsg("Msg %s has no rcvrs or bcast instructions?", msg->ClassName()); hsAssert(!(directCom && bCast), "msg has both rcvrs and bcast instructions, rcvrs ignored"); if (directCom && !bCast) { netMsgWrap->SetBit(plNetMessage::kHasGameMsgRcvrs); // for quick server filtering netMsgWrap->StreamInfo()->SetCompressionType(plNetMessage::kCompressionDont); } // // check for net propagated plasma msgs which should be filtered by relevance regions. // currently only avatar control messages. // if (msg->HasBCastFlag(plMessage::kNetUseRelevanceRegions)) { netMsgWrap->SetBit(plNetMessage::kUseRelevanceRegions); } // // CCRs can route a plMessage to all online players. // hsBool ccrSendToAllPlayers = false; #ifndef PLASMA_EXTERNAL_RELEASE if ( AmCCR() ) { ccrSendToAllPlayers = msg->HasBCastFlag( plMessage::kCCRSendToAllPlayers ); if ( ccrSendToAllPlayers ) netMsgWrap->SetBit( plNetMessage::kRouteToAllPlayers ); } #endif // // check for inter-age routing. if set, online rcvrs not in current age will receive // this msg courtesy of pls routing. // if ( !ccrSendToAllPlayers ) { hsBool allowInterAge = msg->HasBCastFlag( plMessage::kNetAllowInterAge ); if ( allowInterAge ) netMsgWrap->SetBit(plNetMessage::kInterAgeRouting); } // check for reliable send if (msg->HasBCastFlag(plMessage::kNetSendUnreliable) && !(synchedObj && (synchedObj->GetSynchFlags() & plSynchedObject::kSendReliably)) ) netMsgWrap->SetBit(plNetMessage::kNeedsReliableSend, 0); // clear reliable net send bit #ifdef HS_DEBUGGING Int16 type=*(Int16*)netMsgWrap->StreamInfo()->GetStreamBuf(); hsAssert(type>=0 && typeSetPlayerID(GetPlayerID()); netMsgWrap->SetNetProtocol(kNetProtocolCli2Game); int ret = SendMsg(netMsgWrap); if (plNetObjectDebugger::GetInstance()->IsDebugObject(msg->GetSender() ? msg->GetSender()->ObjectIsLoaded() : nil)) { #if 0 hsLogEntry(plNetObjectDebugger::GetInstance()->LogMsg( xtl::format(" object:%s, rcvr %s %s", msg->GetSender(), msg->GetNumReceivers() ? msg->GetReceiver(0)->GetName() : "?", netMsgWrap->AsStdString().c_str()).c_str())); #endif } delete netMsgWrap; return ret; } // // Send a net msg. Delivers to transport mgr who sends p2p or to server // int plNetClientMgr::SendMsg(plNetMessage* msg) { if (GetFlagsBit(kDisabled)) return hsOK; if (!CanSendMsg(msg)) return hsOK; // If we're recording messages, set an identifying flag and echo the message back to ourselves if (fMsgRecorder && fMsgRecorder->IsRecordableMsg(msg)) { msg->SetBit(plNetMessage::kEchoBackToSender, true); } msg->SetTimeSent(plUnifiedTime::GetCurrentTime()); int channel = IPrepMsg(msg); // hsLogEntry( DebugMsg( " %s %s", msg->ClassName(), msg->AsStdString().c_str()) ); int ret=fTransport.SendMsg(channel, msg); // Debug if (plNetMsgVoice::ConvertNoRef(msg)) SetFlagsBit(kSendingVoice); if (plNetMsgGameMessage::ConvertNoRef(msg)) SetFlagsBit(kSendingActions); plCheckNetMgrResult_ValReturn(ret,(char*)xtl::format("Failed to send %s, NC ret=%d", msg->ClassName(), ret).c_str()); return ret; } void plNetClientMgr::StoreSDLState(const plStateDataRecord* sdRec, const plUoid& uoid, UInt32 sendFlags, UInt32 writeOptions) { // send to server plNetMsgSDLState* msg = sdRec->PrepNetMsg(0, writeOptions); msg->SetNetProtocol(kNetProtocolCli2Game); msg->ObjectInfo()->SetUoid(uoid); if (sendFlags & plSynchedObject::kNewState) msg->SetBit(plNetMessage::kNewSDLState); if (sendFlags & plSynchedObject::kUseRelevanceRegions) msg->SetBit(plNetMessage::kUseRelevanceRegions); if (sendFlags & plSynchedObject::kDontPersistOnServer) msg->SetPersistOnServer(false); if (sendFlags & plSynchedObject::kIsAvatarState) msg->SetIsAvatarState(true); bool broadcast = (sendFlags & plSynchedObject::kBCastToClients) != 0; if (broadcast && plNetClientApp::GetInstance()) { msg->SetPlayerID(plNetClientApp::GetInstance()->GetPlayerID()); } SendMsg(msg); DEL(msg); }