X-Git-Url: https://git.librecmc.org/?a=blobdiff_plain;ds=sidebyside;f=src%2Fnodemetadata.cpp;h=b84ffc8cbc2f9230534259fd7cc5670603303097;hb=cb00632e23a41d8d171631de9d85e168b251b80e;hp=9b60cf33e1a39d8d92f461b06528d8f9ba5b193b;hpb=bbdd869d72d7b5aae2994d287f69e1c1d866f4e2;p=oweals%2Fminetest.git diff --git a/src/nodemetadata.cpp b/src/nodemetadata.cpp index 9b60cf33e..b84ffc8cb 100644 --- a/src/nodemetadata.cpp +++ b/src/nodemetadata.cpp @@ -23,6 +23,7 @@ with this program; if not, write to the Free Software Foundation, Inc., #include "inventory.h" #include "log.h" #include "util/serialize.h" +#include "util/basic_macros.h" #include "constants.h" // MAP_BLOCKSIZE #include @@ -39,28 +40,36 @@ NodeMetadata::~NodeMetadata() delete m_inventory; } -void NodeMetadata::serialize(std::ostream &os) const +void NodeMetadata::serialize(std::ostream &os, u8 version, bool disk) const { - int num_vars = m_stringvars.size(); + int num_vars = disk ? m_stringvars.size() : countNonPrivate(); writeU32(os, num_vars); - for (StringMap::const_iterator - it = m_stringvars.begin(); - it != m_stringvars.end(); ++it) { - os << serializeString(it->first); - os << serializeLongString(it->second); + for (const auto &sv : m_stringvars) { + bool priv = isPrivate(sv.first); + if (!disk && priv) + continue; + + os << serializeString(sv.first); + os << serializeLongString(sv.second); + if (version >= 2) + writeU8(os, (priv) ? 1 : 0); } m_inventory->serialize(os); } -void NodeMetadata::deSerialize(std::istream &is) +void NodeMetadata::deSerialize(std::istream &is, u8 version) { - m_stringvars.clear(); + clear(); int num_vars = readU32(is); for(int i=0; i= 2) { + if (readU8(is) == 1) + markPrivate(name, true); + } } m_inventory->deSerialize(is); @@ -69,19 +78,42 @@ void NodeMetadata::deSerialize(std::istream &is) void NodeMetadata::clear() { Metadata::clear(); + m_privatevars.clear(); m_inventory->clear(); } bool NodeMetadata::empty() const { - return Metadata::empty() && m_inventory->getLists().size() == 0; + return Metadata::empty() && m_inventory->getLists().empty(); +} + + +void NodeMetadata::markPrivate(const std::string &name, bool set) +{ + if (set) + m_privatevars.insert(name); + else + m_privatevars.erase(name); +} + +int NodeMetadata::countNonPrivate() const +{ + // m_privatevars can contain names not actually present + // DON'T: return m_stringvars.size() - m_privatevars.size(); + int n = 0; + for (const auto &sv : m_stringvars) { + if (!isPrivate(sv.first)) + n++; + } + return n; } /* NodeMetadataList */ -void NodeMetadataList::serialize(std::ostream &os) const +void NodeMetadataList::serialize(std::ostream &os, u8 blockver, bool disk, + bool absolute_pos) const { /* Version 0 is a placeholder for "nothing to see here; go away." @@ -93,26 +125,33 @@ void NodeMetadataList::serialize(std::ostream &os) const return; } - writeU8(os, 1); // version + u8 version = (blockver > 27) ? 2 : 1; + writeU8(os, version); writeU16(os, count); - for(std::map::const_iterator + for (NodeMetadataMap::const_iterator i = m_data.begin(); - i != m_data.end(); ++i) - { + i != m_data.end(); ++i) { v3s16 p = i->first; NodeMetadata *data = i->second; if (data->empty()) continue; - u16 p16 = p.Z * MAP_BLOCKSIZE * MAP_BLOCKSIZE + p.Y * MAP_BLOCKSIZE + p.X; - writeU16(os, p16); - - data->serialize(os); + if (absolute_pos) { + writeS16(os, p.X); + writeS16(os, p.Y); + writeS16(os, p.Z); + } else { + // Serialize positions within a mapblock + u16 p16 = (p.Z * MAP_BLOCKSIZE + p.Y) * MAP_BLOCKSIZE + p.X; + writeU16(os, p16); + } + data->serialize(os, version, disk); } } -void NodeMetadataList::deSerialize(std::istream &is, IItemDefManager *item_def_mgr) +void NodeMetadataList::deSerialize(std::istream &is, + IItemDefManager *item_def_mgr, bool absolute_pos) { clear(); @@ -123,7 +162,7 @@ void NodeMetadataList::deSerialize(std::istream &is, IItemDefManager *item_def_m return; } - if (version != 1) { + if (version > 2) { std::string err_str = std::string(FUNCTION_NAME) + ": version " + itos(version) + " not supported"; infostream << err_str << std::endl; @@ -132,26 +171,29 @@ void NodeMetadataList::deSerialize(std::istream &is, IItemDefManager *item_def_m u16 count = readU16(is); - for (u16 i=0; i < count; i++) { - u16 p16 = readU16(is); - + for (u16 i = 0; i < count; i++) { v3s16 p; - p.Z = p16 / MAP_BLOCKSIZE / MAP_BLOCKSIZE; - p16 &= MAP_BLOCKSIZE * MAP_BLOCKSIZE - 1; - p.Y = p16 / MAP_BLOCKSIZE; - p16 &= MAP_BLOCKSIZE - 1; - p.X = p16; - + if (absolute_pos) { + p.X = readS16(is); + p.Y = readS16(is); + p.Z = readS16(is); + } else { + u16 p16 = readU16(is); + p.X = p16 & (MAP_BLOCKSIZE - 1); + p16 /= MAP_BLOCKSIZE; + p.Y = p16 & (MAP_BLOCKSIZE - 1); + p16 /= MAP_BLOCKSIZE; + p.Z = p16; + } if (m_data.find(p) != m_data.end()) { - warningstream<<"NodeMetadataList::deSerialize(): " - <<"already set data at position" - <<"("<deSerialize(is); + data->deSerialize(is, version); m_data[p] = data; } } @@ -165,7 +207,7 @@ std::vector NodeMetadataList::getAllKeys() { std::vector keys; - std::map::const_iterator it; + NodeMetadataMap::const_iterator it; for (it = m_data.begin(); it != m_data.end(); ++it) keys.push_back(it->first); @@ -174,7 +216,7 @@ std::vector NodeMetadataList::getAllKeys() NodeMetadata *NodeMetadataList::get(v3s16 p) { - std::map::const_iterator n = m_data.find(p); + NodeMetadataMap::const_iterator n = m_data.find(p); if (n == m_data.end()) return NULL; return n->second; @@ -184,7 +226,8 @@ void NodeMetadataList::remove(v3s16 p) { NodeMetadata *olddata = get(p); if (olddata) { - delete olddata; + if (m_is_metadata_owner) + delete olddata; m_data.erase(p); } } @@ -197,9 +240,10 @@ void NodeMetadataList::set(v3s16 p, NodeMetadata *d) void NodeMetadataList::clear() { - std::map::iterator it; - for (it = m_data.begin(); it != m_data.end(); ++it) { - delete it->second; + if (m_is_metadata_owner) { + NodeMetadataMap::const_iterator it; + for (it = m_data.begin(); it != m_data.end(); ++it) + delete it->second; } m_data.clear(); } @@ -207,7 +251,7 @@ void NodeMetadataList::clear() int NodeMetadataList::countNonEmpty() const { int n = 0; - std::map::const_iterator it; + NodeMetadataMap::const_iterator it; for (it = m_data.begin(); it != m_data.end(); ++it) { if (!it->second->empty()) n++;