You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

356 lines
9.3 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/>.
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==*/
#pragma warning(disable: 4786)
#include "hsTimer.h"
#include "plNetTransport.h"
#include "plNetTransportMember.h"
#include "plNetMessage/plNetMessage.h"
#include "plNetClient/plNetClientMgr.h"
#include <algorithm>
plNetTransport::~plNetTransport()
{
ClearMembers();
}
//
// add a member to the master list if not already there
//
int plNetTransport::AddMember(plNetTransportMember* mbr)
{
if (FindMember(mbr)==-1)
{
fMembers.push_back(mbr);
hsLogEntry( plNetClientMgr::GetInstance()->DebugMsg("Adding member %s", mbr->AsStdString().c_str()) );
plNetClientMgr::GetInstance()->GetListenList()->AddMember(mbr);
plNetClientMgr::GetInstance()->GetTalkList()->AddMember(mbr);
DumpState();
return fMembers.size()-1;
}
return -1;
}
void plNetTransport::IUnSubscribeToAllChannelGrps(plNetTransportMember* mbr)
{
int i;
for( i=mbr->GetNumSubscriptions()-1; i>=0 ; i-- )
{
int chan=mbr->GetSubscription(i);
hsBool ok=UnSubscribeToChannelGrp(mbr, chan);
hsAssert(ok, "can't find supposed subscription to remove");
} // for
}
void plNetTransport::IRemoveMember(plNetTransportMember* mbr)
{
if (!mbr)
return;
hsLogEntry( plNetClientMgr::GetInstance()->DebugMsg("Removing member %s", mbr->AsStdString().c_str()) );
// plNetClientMgr::GetInstance()->GetNetCore()->RemovePeer(mbr->GetPeerID());
plNetClientMgr::GetInstance()->GetTalkList()->RemoveMember(mbr);
plNetClientMgr::GetInstance()->GetListenList()->RemoveMember(mbr);
// remove member from subscription lists
IUnSubscribeToAllChannelGrps(mbr);
plMembersList::iterator it=std::find(fMembers.begin(),fMembers.end(),mbr);
// remove member from master list
fMembers.erase(it);
hsLogEntry( plNetClientMgr::GetInstance()->DebugMsg("Done Removing member %s", mbr->AsStdString().c_str()) );
DumpState();
delete mbr;
}
//
// remove member from master list, and all subscription channels.
// return true on success.
//
hsBool plNetTransport::RemoveMember(int idx)
{
if (idx>=0)
{
plNetTransportMember* mbr=GetMember(idx);
IRemoveMember(mbr);
return true;
}
return false;
}
//
// remove member from master list, and all subscription channels.
// return true on success.
//
hsBool plNetTransport::RemoveMember(plNetTransportMember* mbr)
{
IRemoveMember(mbr);
return true;
}
//
// return array index or -1
//
int plNetTransport::FindMember(const plNetTransportMember* mbr)
{
plMembersList::iterator it = std::find(fMembers.begin(), fMembers.end(), mbr);
return (it==fMembers.end()) ? -1 : (it-fMembers.begin());
}
//
// add this member to the given channel grp
//
void plNetTransport::SubscribeToChannelGrp(plNetTransportMember* mbr, int channel)
{
// hsAssert(FindMember(mbr) != -1, "must add member before subscribing to channel");
if (mbr->AddSubscription(channel))
{
hsAssert(channel<fChannelGroups.size(), "invalid channel index");
fChannelGroups[channel].push_back(mbr);
}
}
//
// Remove the subscription to the given channel grp for a member.
//
hsBool plNetTransport::UnSubscribeToChannelGrp(plNetTransportMember* mbr, int chan)
{
hsAssert(chan>=0 && chan<fChannelGroups.size(), "invalid channel idx");
plMembersList* mList = &fChannelGroups[chan];
plMembersList::iterator it=std::find(mList->begin(), mList->end(), mbr);
if (it != mList->end())
{
mList->erase(it);
hsBool ret=mbr->RemoveSubscription(chan);
hsAssert(ret, "error removing subscription");
return true;
}
return false;
}
//
// copy list of members channelGrp subscriptions
//
void plNetTransport::GetSubscriptions(plNetTransportMember* mbr, std::vector<int>* channels) const
{
mbr->CopySubscriptions(channels);
}
//
// Send Msg to all members in the given channelGrp.
// Here's where multicasting would be used.
// Returns neg number (NetCore::RetCode) on send error, 1, if not sent, and 0 if sent
//
int plNetTransport::SendMsg(int chan, plNetMessage* netMsg) const
{
NetCommSendMsg(netMsg);
return hsOK;
plNetClientMgr* nc=plNetClientMgr::GetInstance();
int ret=1; // didn't send
if (chan < fChannelGroups.size())
{
const plMembersList* mList = &fChannelGroups[chan];
// does this msg have a list of receivers
plNetMsgReceiversListHelper* rl = plNetMsgReceiversListHelper::ConvertNoRef(netMsg);
#if 0
// send msg to all subscribers to this channel
int size=mList->size();
for( int i=0 ; i<size; i++ )
{
hsAssert(false, "eric, port me");
plNetTransportMember* tm=(*mList)[i];
hsAssert(tm, "nil mbr in sendMsg");
// int peerID=tm->GetPeerID();
// hsAssert(peerID>=0, "queing message to invalid peer");
// if ((ncRet=nc->GetNetClientComm().SendMsg(netMsg, peerID, sendFlags, msgSize)) != plNetCore::kNetOK)
NetCommSendMsg(netMsg);
if (rl)
{
hsBool ok=rl->RemoveReceiverPlayerID(tm->GetPlayerID());
hsAssert(ok, "couldn't find rcvr to remove?");
}
ret=0; // sent ok
} // for
#endif
// if there are rcvrs left that we couldn't send to, send via server
if (rl && rl->GetNumReceivers())
{
// if ((ncRet=nc->GetNetClientComm().SendMsg(netMsg, nc->GetServerPeerID(), sendFlags, msgSize)) != plNetCore::kNetOK)
NetCommSendMsg(netMsg);
ret=0; // sent
}
}
else
{
hsStatusMessage("EMPTY TRANSPORT GROUP\n");
}
return ret;
}
void plNetTransport::ClearMembers()
{
int i;
for( i=0 ;i<GetNumMembers() ;i++ )
{
plNetTransportMember* mbr = GetMember(i);
hsAssert(mbr, "nil member?");
IUnSubscribeToAllChannelGrps(mbr);
delete mbr;
} // for
fMembers.clear();
}
//
// return array index or -1
//
int plNetTransport::FindMember(UInt32 playerID) const
{
int i;
for( i=0 ;i<GetNumMembers() ;i++ )
{
plNetTransportMember* mbr = GetMember(i);
if (mbr->GetPlayerID()==playerID)
return i;
}
return -1;
}
//
// return array index or -1
//
int plNetTransport::FindMember(const plKey avKey) const
{
int i;
for( i=0 ;i<GetNumMembers() ;i++ )
{
plNetTransportMember* mbr = GetMember(i);
if (mbr->GetAvatarKey()==avKey)
return i;
}
return -1;
}
//
// clear channel and unsubscribe all members to that channel
//
void plNetTransport::ClearChannelGrp(int channel)
{
const plMembersList* mList = &fChannelGroups[channel];
int i, size=mList->size();
for( i=0 ; i<size; i++ )
{
plNetTransportMember* tm=(*mList)[i];
hsBool ok=tm->RemoveSubscription(channel);
hsAssert(ok, "error removing subscription");
}
fChannelGroups[channel].clear();
}
void plNetTransport::DumpState()
{
plNetClientMgr* nc=plNetClientMgr::GetInstance();
hsLogEntry( nc->DebugMsg("-------------------\n") );
hsLogEntry( nc->DebugMsg("Num Channels=%d\n", fChannelGroups.size()) );
int i;
for(i=0;i<fChannelGroups.size();i++)
{
plMembersList* mList = &fChannelGroups[i];
hsLogEntry( nc->DebugMsg("\tChannel %d, num mbrs=%d\n", i, mList->size()) );
int j;
for(j=0; j<mList->size();j++)
{
plNetTransportMember * mbr = (*mList)[j];
hsLogEntry( nc->DebugMsg("\t\tMbr %s\n",(*mList)[j]->AsStdString().c_str()) );
}
}
nc->DebugMsg("Num Mbrs=%d\n", GetNumMembers());
for(i=0;i<GetNumMembers();i++)
{
plNetTransportMember * mbr = GetMember(i);
hsLogEntry (nc->DebugMsg("\tMbr %d, name=%s, plyrID=%lu, subs=%d",
i,mbr->AsStdString().c_str(),mbr->GetPlayerID(),mbr->GetNumSubscriptions()) );
int j;
for(j=0;j<mbr->GetNumSubscriptions();j++)
{
hsLogEntry( nc->DebugMsg("\t\tSub %d, chan=%d\n", j, mbr->GetSubscription(j)) );
}
}
hsLogEntry( nc->DebugMsg("\n") );
}
void plNetTransport::SetNumChannels(int n)
{
if (n>fChannelGroups.size())
fChannelGroups.resize(n);
}
int compare( const void* arg1, const void *arg2 )
{
plNetTransportMember** m1 = (plNetTransportMember**)arg1;
plNetTransportMember** m2 = (plNetTransportMember**)arg2;
float d1=m1 ? (*m1)->GetDistSq() : hsScalarMax;
float d2=m2 ? (*m2)->GetDistSq() : hsScalarMax;
return (int)(d1-d2);
}
//
// create a members list sorted by dist.
// caller must delete this when done
//
void plNetTransport::GetMemberListDistSorted(plNetTransportMember**& listIn) const
{
// copy members list
listIn = TRACKED_NEW plNetTransportMember* [fMembers.size()];
int i;
for (i=0; i<fMembers.size(); i++)
listIn[i]=fMembers[i];
// sort members list
qsort(listIn, fMembers.size(), sizeof(plNetTransportMember*), compare);
}