// Ryzom - MMORPG Framework
// Copyright (C) 2010 Winch Gate Property Limited
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero 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 Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see .
#include "stdpch.h"
#include "state_instance.h"
#include "nel/misc/smart_ptr.h"
#include "script_vm.h"
#include "script_compiler.h"
#include "event_reaction_container.h"
#include "ai_grp.h"
#include "ai_outpost.h"
using namespace std;
using namespace NLMISC;
using namespace AICOMP;
using namespace AIVM;
//////////////////////////////////////////////////////////////////////////////
// CStateInstance //
//////////////////////////////////////////////////////////////////////////////
CAIState const* CStateInstance::getActiveState() const
{
#if !FINAL_VERSION
nlassert(this!=NULL);
#else
if (!this)
return NULL; // ugly! -> to remove date = 19/05/2004.
#endif
if (_PunctualState) // the puntual state is active !
return _PunctualState;
if (_state) // the normal state is active !
return _state;
// no state is active !
return NULL;
}
sint32 CStateInstance::getUserTimer(uint timerId)
{
NLMISC::clamp(timerId, 0u, 3u);
return _UserTimer[timerId].timeRemaining();
}
void CStateInstance::setUserTimer(uint timerId, sint32 time)
{
NLMISC::clamp(timerId, 0u, 3u);
if (_UserTimer[timerId].isEnabled())
_UserTimer[timerId].set(time);
}
void CStateInstance::addStatePersistentObj(CAIState const* keyState, CSmartPtr anyObj)
{
_StatePersistentObjList.push_back(CStatePersistentObjEntry(keyState, anyObj));
}
void CStateInstance::setFirstBotSpawned()
{
_FirstBotSpawned = true;
}
struct CDiffStateRemover
{
CDiffStateRemover(CAIState const* const state) : _State(state) { }
bool operator()(CStateInstance::CStatePersistentObjEntry const& entry) const
{
return entry._State!=_State;
}
CAIState const* const _State;
};
void CStateInstance::removeExceptForState(CAIState const* keyState)
{
TStatePersistentObjList::iterator const itLast = remove_if(_StatePersistentObjList.begin(), _StatePersistentObjList.end(), CDiffStateRemover(keyState));
_StatePersistentObjList.erase(itLast, _StatePersistentObjList.end());
}
void CStateInstance::setScriptCallBack(TStringId const& eventName, AIVM::CByteCodeEntry const& codeScriptEntry)
{
_CallBacks[eventName] = codeScriptEntry; // effectively affects this callback to this eventName.
}
AIVM::CByteCodeEntry const* CStateInstance::getScriptCallBackPtr(TStringId const& eventName) const
{
TCallBackList::const_iterator it = _CallBacks.find(eventName);
if (it==_CallBacks.end())
return NULL;
return &(it->second);
}
void CStateInstance::callScriptCallBack(IScriptContext* caller, TStringId const& funcName, int mode, std::string const& inParamsSig, std::string const& outParamsSig, AIVM::CScriptStack* stack)
{
// :FIXME: mode, signatures end stack are unused
AIVM::CByteCodeEntry const* const codeScript = getScriptCallBackPtr(funcName);
if (!codeScript || !codeScript->isValid())
return;
interpretCode(caller, *codeScript); // effectively calls the call back.
}
void CStateInstance::callNativeCallBack(IScriptContext* caller, std::string const& funcName, int mode, std::string const& inParamsSig, std::string const& outParamsSig, AIVM::CScriptStack* stack)
{
// :FIXME: caller is unused
CScriptNativeFuncParams* funcParam = CCompiler::getNativeFunc(funcName, inParamsSig, outParamsSig);
if (funcParam!=NULL)
{
bool varArg = (mode & 1)!=0; // :KLUDGE: Hardcoded 1 :TODO: Replace with a names constant (see script_compiler.cpp)
if (stack!=NULL)
{
if (varArg)
{
stack->push(outParamsSig);
stack->push(inParamsSig);
}
funcParam->_func(this, *stack);
}
else
{
AIVM::CScriptStack dummyStack;
if (varArg)
{
dummyStack.push(outParamsSig);
dummyStack.push(inParamsSig);
}
funcParam->_func(this, dummyStack);
}
}
else
{
nlwarning("Trying to call an unknown native function '%s_%s_%s'", funcName.c_str(), inParamsSig.c_str(), outParamsSig.c_str());
// TODO:kxu: rebuild stack
}
}
void CStateInstance::dumpVarsAndFunctions(CStringWriter& sw) const
{
sw.append("float variables:");
FOREACHC(varIt, TLogicVarList, _LogicVar)
sw.append(" "+CStringMapper::unmap(varIt->first)+" = "+NLMISC::toString(varIt->second));
sw.append("string variables:");
FOREACHC(varIt, TStrLogicVarList, _StrLogicVar)
sw.append(" "+CStringMapper::unmap(varIt->first)+" = "+varIt->second);
sw.append("context variables:");
FOREACHC(varIt, TCtxLogicVarList, _CtxLogicVar)
sw.append(" "+CStringMapper::unmap(varIt->first)+" = "+(int)(void*)varIt->second);
sw.append("callBacks:");
FOREACHC(varIt, TCallBackList, _CallBacks)
sw.append(" "+CStringMapper::unmap(varIt->first));
}
void CStateInstance::interpretCode(IScriptContext* caller, CByteCodeEntry const& codeScriptEntry)
{
AIVM::IScriptContext* parent = getPersistentStateInstance()->getParentStateInstance();
AIVM::CScriptVM::getInstance()->interpretCode(this, parent, caller, codeScriptEntry);
}
void CStateInstance::interpretCode(IScriptContext* caller, CSmartPtr const& codeScript)
{
interpretCode(caller, CByteCodeEntry(codeScript, 0));
}
void CStateInstance::interpretCodeOnChildren(CByteCodeEntry const& codeScriptEntry)
{
vector > tmpList;
FOREACH(childIt, CPersistentStateInstance::TChildList, getPersistentStateInstance()->childs())
tmpList.push_back((*childIt)->getGroup());
FOREACH(childIt, vector >, tmpList)
(*childIt)->getPersistentStateInstance()->interpretCode(this, codeScriptEntry);
}
IScriptContext* CStateInstance::findContext(NLMISC::TStringId const strId)
{
#if !FINAL_VERSION
nlassert(this->getGroup());
#endif
std::vector grps;
this->getGroup()->getAIInstance()->findGroup(grps, CStringMapper::unmap(strId));
if (grps.size()==1)
return grps.back()->getPersistentStateInstance();
else
return NULL;
}
std::string CStateInstance::getContextName()
{
return getGroup()->getFullName();
}
void CStateInstance::init(CAIState* startState)
{
_state=
_PunctualState=
_NextPunctualState=NULL;
_CancelPunctualState = false;
_NextState = startState;
_LogicVarChanged = false;
_FirstBotSpawned = false;
// _VarIndex[CStringMapper::map("v0")]=0;
// _VarIndex[CStringMapper::map("v1")]=1;
// _VarIndex[CStringMapper::map("v2")]=2;
// _VarIndex[CStringMapper::map("v3")]=3;
}
void CStateInstance::setGlobalNelVar(std::string const& varId, float value)
{
if (NLMISC::ICommand::exists(varId))
NLMISC::ICommand::execute(NLMISC::toString("%s %f", varId.c_str(), value), *NLMISC::InfoLog);
else
nlwarning("Nel variable \"%s\" do not exist", varId.c_str());
}
void CStateInstance::setGlobalNelVar(std::string const& varId, std::string value)
{
std::vector args; args.push_back(value);
if (NLMISC::ICommand::exists(varId))
NLMISC::ICommand::execute(NLMISC::toString("%s %s", varId.c_str(), value.c_str()), *NLMISC::InfoLog);
else
nlwarning("Nel variable \"%s\" do not exist", varId.c_str());
}
void CStateInstance::blockUserEvent(uint32 eventId)
{
_UserEventBlocked |= (1 << eventId);
}
void CStateInstance::unblockUserEvent(uint32 eventId)
{
_UserEventBlocked &= ~(1 << eventId);
}
bool CStateInstance::isUserEventBlocked(uint32 eventId) const
{
return ((1 << eventId) & _UserEventBlocked) != 0;
}
//////////////////////////////////////////////////////////////////////////////
// CPersistentStateInstance //
//////////////////////////////////////////////////////////////////////////////
void CPersistentStateInstance::updateStateInstance()
{
// deal with timer events -----------------------------------------
// Event: State Timeout
if (_StateTimeout.testOnce())
processStateEvent(getEventContainer().EventPositionalStateTimeout);
// Event: Punctual State Timeout
if (_PunctualStateTimeout.testOnce())
processStateEvent(getEventContainer().EventPunctualStateTimeout);
// Event: User Timer n Triggered
for (uint i=0;i<4;++i)
{
if (!_UserTimer[i].testOnce())
continue;
processStateEvent(getEventContainer().EventUserTimer[i]);
}
// Event logic var changed
if (_LogicVarChanged)
{
processStateEvent(getEventContainer().EventVariableChanged);
for(uint i =0;i<4;++i)
{
if(_LogicVarChangedList[i]==true)
{
processStateEvent(getEventContainer().EventVariableChangedTab[i]);
_LogicVarChangedList[i]=false;
}
}
_LogicVarChanged = false;
}
// deal with positional state change ------------------------------
// if state change requested (and not in punctual state) then setup new state
if (_NextState)
{
CAIState const* oldState = NULL;
if (!_PunctualState && !_NextPunctualState && _state)
{
// close down current state
oldState = _state;
processStateEvent(getEventContainer().EventEndOfState);
}
// switch state & initalise state status flags
_state = _NextState;
_NextState = NULL;
_StateTimeout.disable();
if (_PunctualState || _NextPunctualState)
return;
removeExceptForState(_state);
processStateEvent(getEventContainer().EventStartOfState);
stateChange(oldState, _state);
}
// deal with start of punctual state ------------------------------
// if state change requested then setup new state
if (_NextPunctualState)
{
CAIState const* oldState = NULL;
breakable
{
if (_PunctualState)
{
oldState = _PunctualState;
break;
}
if (_state)
{
oldState = _state;
break;
}
}
// switch state & initalize state status flags
_PunctualState = _NextPunctualState;
_NextPunctualState = NULL;
_CancelPunctualState = false;
_PunctualStateTimeout.disable();
// initalize new state
CAIState const* const newState = _PunctualState;
removeExceptForState(newState);
processStateEvent(getEventContainer().EventStartOfState, newState);
stateChange(oldState, newState);
}
// must be done after start of state
if (_FirstBotSpawned)
{
_FirstBotSpawned = false;
processStateEvent(getEventContainer().EventFirstBotSpawned);
}
// deal with end of punctual state --------------------------------
if (!_CancelPunctualState)
return;
_CancelPunctualState = false;
// if there was an active punctual state then cancel it
if (!_PunctualState)
return;
// switch state & initalise state status flags
_PunctualStateTimeout.disable();
_StateTimeout.resume(); // this line just in case timeout suspended during punctual state
// close down current state
CAIState const* const punctualState = _PunctualState;
processStateEvent(getEventContainer().EventEndOfState, punctualState);
_PunctualState = NULL;
// this removes obj that depends on state affectation existence.
removeExceptForState(_state);
// specialized virtual state change call.
stateChange(punctualState, _state);
}
#include "event_reaction_include.h"