test-client-godot/modules/referentialmessage/referentialmessagecore.cpp
2020-05-02 20:52:28 +02:00

184 lines
No EOL
5.5 KiB
C++

/*
ReferentialMessageCore : Class to store referential to decode message
Copyright (C) 2019 AleaJactaEst
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/>.
*/
#include "modules/debug/debug.h"
#include "modules/impulse/impulsebase.h"
#include "referentialmessagecore.h"
ReferentialMessageCore *ReferentialMessageCore::singleton = nullptr;
/*
* ElementReferential
*/
void ElementReferential::show(int level, int pos)
{
String a;
for(int i=0;i<level;++i)
a += String(CharString("+"));
if(_id == ImpulseBase::Impulse::__LAST_ELEMENT)
{
//OS::get_singleton()->print("[%s:%d] level:%d %s pos:%3d ------ power:%d\n", __FILE__, __LINE__, level, a.ascii().get_data(), pos, _power2);
DBG_PRINT("level:" + itos(level) + " " + a + " id:" + ImpulseBase::get_command_name(_id) + " pos:" + itos(pos) + " ------ power:" + itos(_power2));
}
else
{
DBG_PRINT("level:" + itos(level) + " " + a + " pos:" + itos(pos) + " id:" + ImpulseBase::get_command_name(_id));
}
level ++;
for(int i=0;i<_children.size();++i)
{
Ref<ElementReferential> child = _children[i];
child->show(level, i);
}
}
int ElementReferential::read_command(Ref<BitStream> msgin)
{
uint32_t i = msgin->get_serial(_power2);
Ref<ElementReferential> child = _children[i];
if ( child->_id != ImpulseBase::Impulse::__LAST_ELEMENT )
return child->_id;
return child->read_command(msgin);
}
/*
* ReferentialMessageCore
*/
void ReferentialMessageCore::_bind_methods()
{
}
ReferentialMessageCore *ReferentialMessageCore::get_singleton()
{
if ( ReferentialMessageCore::singleton == nullptr )
ReferentialMessageCore::singleton = new ReferentialMessageCore();
return ReferentialMessageCore::singleton;
}
void ReferentialMessageCore::read_referential_step(Dictionary step, Vector<ElementHead> head, Ref<ElementReferential> root)
{
uint32_t i;
Array keys = step.keys();
String name;
String sep = ":";
if ( keys.size() == 0 )
{
Ref<BitStream> ele;
String a;
int ii;
ele.instance();
for(ii=0;ii<head.size() - 1;++ii)
{
a += head.get(ii).get_name() + String(CharString("_"));
uint32_t _size = head.get(ii).get_size();
ele->put_serial(head.get(ii).get_pos(), getPowerOf2(_size));
}
a += head.get(ii).get_name();
uint32_t _size = head.get(ii).get_size()+1;
ele->put_serial(head.get(ii).get_pos(), getPowerOf2(_size));
int id = ImpulseBase::get_command(a);
if (id != ImpulseBase::Impulse::__LAST_ELEMENT)
{
_encoder[id].copy_ref_bitstream(ele);
root->set_id(id);
}
ele.unref();
return;
}
root->set_size(keys.size());
uint32_t nb = keys.size() - 1;
String a;
for(i=0; i <= nb; ++i)
{
name = keys[i];
Vector<String> info = name.split(sep, false, 2);
if(info.size() == 2 )
{
uint32_t id = info[0].to_int();
Ref<ElementReferential> root_child = root->add_child(ImpulseBase::Impulse::__LAST_ELEMENT, i);
Variant child ;
child = step[name];
ElementHead tmp;
tmp.create(id, nb, info[1].ascii().get_data());
head.push_back(tmp);
read_referential_step(child, head, root_child);
head.remove(head.size()-1);
}
}
}
void ReferentialMessageCore::read_referential(Dictionary dictionary_message)
{
Vector<ElementHead> head;
DBG_PRINT("Read Referential Message");
_decoder->set_id(ImpulseBase::Impulse::__LAST_ELEMENT);
read_referential_step(dictionary_message, head, _decoder);
}
void ReferentialMessageCore::show()
{
// For fun, list powerOf2 - TODO check is same with server side, and check size is same value for msg.xml
/*
{
for(int i=0;i<67;++i)
DBG_PRINT(itos(i) + "=> powerOf2 = " + itos(getPowerOf2(i)));
}
*/
// show Element to push data
{
for(int i = 0; i < ImpulseBase::Impulse::__LAST_ELEMENT; ++i)
{
DBG_PRINT(itos(i) + ") " + ImpulseBase::get_command_name(i) + " " + this->_encoder[i].show().ascii().get_data() + " [size:" + itos(this->_encoder[i].size_data()) + "]");
}
}
// show element to read data
_decoder->show();
}
void ReferentialMessageCore::clear_session()
{
if ( ReferentialMessageCore::singleton == nullptr )
return;
DBG_PRINT("Clear session for Referential Message.");
delete ReferentialMessageCore::singleton;
ReferentialMessageCore::singleton = nullptr;
}
int ReferentialMessageCore::read_command(Ref<BitStream> msgin)
{
return _decoder->read_command(msgin);
}
void ReferentialMessageCore::write_command(ImpulseBase::Impulse command, Ref<BitStream> msgout)
{
#ifdef DEBUG_ENABLED
if ( command >= ImpulseBase::Impulse::__LAST_ELEMENT )
{
ERR_PRINT("Out of range (" + itos(command) + " / " + itos(ImpulseBase::Impulse::__LAST_ELEMENT) + ")");
throw "Out of range";
}
#endif
msgout->put_bitstream(_encoder[command]);
}