2012-11-26 10:18:34 +01:00
|
|
|
/*
|
2013-02-24 18:40:43 +01:00
|
|
|
Minetest
|
2013-02-24 19:38:45 +01:00
|
|
|
Copyright (C) 2010-2013 celeron55, Perttu Ahola <celeron55@gmail.com>
|
2012-11-26 10:18:34 +01:00
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU Lesser General Public License as published by
|
|
|
|
the Free Software Foundation; either version 2.1 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 Lesser General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU Lesser General Public License along
|
|
|
|
with this program; if not, write to the Free Software Foundation, Inc.,
|
|
|
|
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "staticobject.h"
|
|
|
|
#include "util/serialize.h"
|
2020-04-11 11:22:15 +02:00
|
|
|
#include "server/serveractiveobject.h"
|
2018-03-09 08:25:48 +01:00
|
|
|
|
|
|
|
StaticObject::StaticObject(const ServerActiveObject *s_obj, const v3f &pos_):
|
|
|
|
type(s_obj->getType()),
|
|
|
|
pos(pos_)
|
|
|
|
{
|
|
|
|
s_obj->getStaticData(&data);
|
|
|
|
}
|
2012-11-26 10:18:34 +01:00
|
|
|
|
2022-06-08 19:33:46 +02:00
|
|
|
void StaticObject::serialize(std::ostream &os) const
|
2012-11-26 10:18:34 +01:00
|
|
|
{
|
|
|
|
// type
|
2014-11-17 03:52:24 +01:00
|
|
|
writeU8(os, type);
|
2012-11-26 10:18:34 +01:00
|
|
|
// pos
|
2022-06-08 19:33:46 +02:00
|
|
|
writeV3F1000(os, clampToF1000(pos));
|
2012-11-26 10:18:34 +01:00
|
|
|
// data
|
2020-09-20 13:12:55 +02:00
|
|
|
os<<serializeString16(data);
|
2012-11-26 10:18:34 +01:00
|
|
|
}
|
2021-08-19 20:14:22 +02:00
|
|
|
|
2012-11-26 10:18:34 +01:00
|
|
|
void StaticObject::deSerialize(std::istream &is, u8 version)
|
|
|
|
{
|
|
|
|
// type
|
2014-11-17 03:52:24 +01:00
|
|
|
type = readU8(is);
|
2012-11-26 10:18:34 +01:00
|
|
|
// pos
|
2014-11-17 03:52:24 +01:00
|
|
|
pos = readV3F1000(is);
|
2012-11-26 10:18:34 +01:00
|
|
|
// data
|
2020-09-20 13:12:55 +02:00
|
|
|
data = deSerializeString16(is);
|
2012-11-26 10:18:34 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void StaticObjectList::serialize(std::ostream &os)
|
|
|
|
{
|
2021-08-19 20:14:22 +02:00
|
|
|
// Check for problems first
|
|
|
|
auto problematic = [] (StaticObject &obj) -> bool {
|
|
|
|
if (obj.data.size() > U16_MAX) {
|
|
|
|
errorstream << "StaticObjectList::serialize(): "
|
|
|
|
"object has excessive static data (" << obj.data.size() <<
|
|
|
|
"), deleting it." << std::endl;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
};
|
|
|
|
for (auto it = m_stored.begin(); it != m_stored.end(); ) {
|
|
|
|
if (problematic(*it))
|
|
|
|
it = m_stored.erase(it);
|
|
|
|
else
|
|
|
|
it++;
|
|
|
|
}
|
|
|
|
for (auto it = m_active.begin(); it != m_active.end(); ) {
|
|
|
|
if (problematic(it->second))
|
|
|
|
it = m_active.erase(it);
|
|
|
|
else
|
|
|
|
it++;
|
|
|
|
}
|
|
|
|
|
2012-11-26 10:18:34 +01:00
|
|
|
// version
|
2014-11-17 03:52:24 +01:00
|
|
|
u8 version = 0;
|
|
|
|
writeU8(os, version);
|
2015-09-29 01:55:12 +02:00
|
|
|
|
2012-11-26 10:18:34 +01:00
|
|
|
// count
|
2015-09-29 01:55:12 +02:00
|
|
|
size_t count = m_stored.size() + m_active.size();
|
|
|
|
// Make sure it fits into u16, else it would get truncated and cause e.g.
|
|
|
|
// issue #2610 (Invalid block data in database: unsupported NameIdMapping version).
|
2015-10-04 08:50:04 +02:00
|
|
|
if (count > U16_MAX) {
|
2015-09-29 01:55:12 +02:00
|
|
|
errorstream << "StaticObjectList::serialize(): "
|
|
|
|
<< "too many objects (" << count << ") in list, "
|
|
|
|
<< "not writing them to disk." << std::endl;
|
|
|
|
writeU16(os, 0); // count = 0
|
|
|
|
return;
|
|
|
|
}
|
2014-11-17 03:52:24 +01:00
|
|
|
writeU16(os, count);
|
2015-09-29 01:55:12 +02:00
|
|
|
|
2017-08-20 13:30:50 +02:00
|
|
|
for (StaticObject &s_obj : m_stored) {
|
2012-11-26 10:18:34 +01:00
|
|
|
s_obj.serialize(os);
|
|
|
|
}
|
2017-08-20 13:30:50 +02:00
|
|
|
|
|
|
|
for (auto &i : m_active) {
|
|
|
|
StaticObject s_obj = i.second;
|
2012-11-26 10:18:34 +01:00
|
|
|
s_obj.serialize(os);
|
|
|
|
}
|
|
|
|
}
|
2022-11-03 17:35:31 +01:00
|
|
|
|
2012-11-26 10:18:34 +01:00
|
|
|
void StaticObjectList::deSerialize(std::istream &is)
|
|
|
|
{
|
2016-12-11 18:47:50 +01:00
|
|
|
if (m_active.size()) {
|
|
|
|
errorstream << "StaticObjectList::deSerialize(): "
|
|
|
|
<< "deserializing objects while " << m_active.size()
|
|
|
|
<< " active objects already exist (not cleared). "
|
|
|
|
<< m_stored.size() << " stored objects _were_ cleared"
|
|
|
|
<< std::endl;
|
|
|
|
}
|
|
|
|
m_stored.clear();
|
|
|
|
|
2012-11-26 10:18:34 +01:00
|
|
|
// version
|
2014-11-17 03:52:24 +01:00
|
|
|
u8 version = readU8(is);
|
2012-11-26 10:18:34 +01:00
|
|
|
// count
|
2014-11-17 03:52:24 +01:00
|
|
|
u16 count = readU16(is);
|
2015-03-04 17:18:57 +01:00
|
|
|
for(u16 i = 0; i < count; i++) {
|
2012-11-26 10:18:34 +01:00
|
|
|
StaticObject s_obj;
|
|
|
|
s_obj.deSerialize(is, version);
|
|
|
|
m_stored.push_back(s_obj);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-11-03 17:35:31 +01:00
|
|
|
bool StaticObjectList::storeActiveObject(u16 id)
|
|
|
|
{
|
|
|
|
const auto i = m_active.find(id);
|
|
|
|
if (i == m_active.end())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
m_stored.push_back(i->second);
|
|
|
|
m_active.erase(id);
|
|
|
|
return true;
|
|
|
|
}
|