Generate MT 0.4 character.b3d files at load time
These generated files are kept in RAM and not on disk.
This commit is contained in:
parent
26a5aed0c4
commit
dae831a223
Binary file not shown.
@ -369,6 +369,8 @@ void set_default_settings()
|
|||||||
|
|
||||||
|
|
||||||
// Server
|
// Server
|
||||||
|
settings->setDefault("compat_player_model", "character.b3d,3d_armor_character.b3d,skinsdb_3d_armor_character_5.b3d");
|
||||||
|
settings->setDefault("compat_send_original_model", "true");
|
||||||
settings->setDefault("disable_escape_sequences", "false");
|
settings->setDefault("disable_escape_sequences", "false");
|
||||||
settings->setDefault("strip_color_codes", "false");
|
settings->setDefault("strip_color_codes", "false");
|
||||||
#if USE_PROMETHEUS
|
#if USE_PROMETHEUS
|
||||||
|
142
src/server.cpp
142
src/server.cpp
@ -418,6 +418,17 @@ void Server::init()
|
|||||||
|
|
||||||
m_modmgr->loadMods(m_script);
|
m_modmgr->loadMods(m_script);
|
||||||
|
|
||||||
|
// m_compat_player_models is used to prevent constant re-parsing of the
|
||||||
|
// setting
|
||||||
|
std::string player_models = g_settings->get("compat_player_model");
|
||||||
|
player_models.erase(std::remove_if(player_models.begin(),
|
||||||
|
player_models.end(), static_cast<int(*)(int)>(&std::isspace)),
|
||||||
|
player_models.end());
|
||||||
|
if (player_models.empty() || isSingleplayer())
|
||||||
|
FATAL_ERROR_IF(!m_compat_player_models.empty(), "Compat player models list not empty");
|
||||||
|
else
|
||||||
|
m_compat_player_models = str_split(player_models, ',');
|
||||||
|
|
||||||
// Read Textures and calculate sha1 sums
|
// Read Textures and calculate sha1 sums
|
||||||
fillMediaCache();
|
fillMediaCache();
|
||||||
|
|
||||||
@ -2495,6 +2506,76 @@ bool Server::SendBlock(session_t peer_id, const v3s16 &blockpos)
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Hacks because I don't want to make duplicate read/write functions for little
|
||||||
|
// endian numbers.
|
||||||
|
u32 readU32_le(std::istream &is) {
|
||||||
|
char buf[4] = {0};
|
||||||
|
is.read(buf, sizeof(buf));
|
||||||
|
std::reverse(buf, buf + sizeof(buf));
|
||||||
|
return readU32((u8 *)buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
v3f readV3F32_le(std::istream &is) {
|
||||||
|
char buf[12] = {0};
|
||||||
|
is.read(buf, sizeof(buf));
|
||||||
|
std::reverse(buf, buf + 4);
|
||||||
|
std::reverse(buf + 4, buf + 8);
|
||||||
|
std::reverse(buf + 8, buf + 12);
|
||||||
|
return readV3F32((u8 *)buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
void writeV3F32_le(std::ostream &os, v3f pos) {
|
||||||
|
char buf[12];
|
||||||
|
writeV3F32((u8 *)buf, pos);
|
||||||
|
std::reverse(buf, buf + 4);
|
||||||
|
std::reverse(buf + 4, buf + 8);
|
||||||
|
std::reverse(buf + 8, buf + 12);
|
||||||
|
os.write(buf, sizeof(buf));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Converts MT 5+ player models into MT 0.4 compatible models
|
||||||
|
std::string makeCompatPlayerModel(std::string b3d) {
|
||||||
|
std::stringstream ss(b3d);
|
||||||
|
|
||||||
|
// ss.read(4) != "BB3D"
|
||||||
|
const u32 header = readU32_le(ss);
|
||||||
|
if (header != 0x44334242) {
|
||||||
|
warningstream << "Invalid B3D header in player model: " << header << std::endl;
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
readU32(ss); // Length
|
||||||
|
readU32(ss); // Version
|
||||||
|
|
||||||
|
// Look for the node
|
||||||
|
while (ss.good()) {
|
||||||
|
const u32 name = readU32_le(ss);
|
||||||
|
const u32 length = readU32_le(ss);
|
||||||
|
|
||||||
|
// name != "NODE"
|
||||||
|
if (name != 0x45444f4e) {
|
||||||
|
ss.ignore(length);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Node name
|
||||||
|
ss.ignore(length, '\x00');
|
||||||
|
|
||||||
|
// Node position
|
||||||
|
std::streampos p = ss.tellg();
|
||||||
|
const v3f offset_pos = readV3F32_le(ss) - v3f(0, BS, 0);
|
||||||
|
|
||||||
|
// Write the new position back to the stringstream
|
||||||
|
ss.seekp(p);
|
||||||
|
writeV3F32_le(ss, offset_pos);
|
||||||
|
|
||||||
|
return ss.str();
|
||||||
|
}
|
||||||
|
|
||||||
|
warningstream << "Could not find base position in B3D file" << std::endl;
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
bool Server::addMediaFile(const std::string &filename,
|
bool Server::addMediaFile(const std::string &filename,
|
||||||
const std::string &filepath, std::string *filedata_to,
|
const std::string &filepath, std::string *filedata_to,
|
||||||
std::string *digest_to)
|
std::string *digest_to)
|
||||||
@ -2548,6 +2629,32 @@ bool Server::addMediaFile(const std::string &filename,
|
|||||||
// Put in list
|
// Put in list
|
||||||
m_media[filename] = MediaInfo(filepath, sha1_base64);
|
m_media[filename] = MediaInfo(filepath, sha1_base64);
|
||||||
|
|
||||||
|
// Add a compatibility model if required
|
||||||
|
if (isCompatPlayerModel(filename)) {
|
||||||
|
// Offset the mesh
|
||||||
|
const std::string filedata_compat = makeCompatPlayerModel(filedata);
|
||||||
|
if (filedata_compat != "") {
|
||||||
|
SHA1 sha1;
|
||||||
|
sha1.addBytes(filedata_compat.c_str(), filedata_compat.length());
|
||||||
|
unsigned char *digest = sha1.getDigest();
|
||||||
|
std::string sha1_base64 = base64_encode(digest, 20);
|
||||||
|
free(digest);
|
||||||
|
|
||||||
|
// If the original model is being sent then rename the
|
||||||
|
// compatibility one so it doesn't conflict. The renamed model is
|
||||||
|
// used in player_sao.cpp if the setting is enabled.
|
||||||
|
std::string fn_compat = filename;
|
||||||
|
if (g_settings->getBool("compat_send_original_model")) {
|
||||||
|
fn_compat = "_mc_compat_" + fn_compat;
|
||||||
|
|
||||||
|
// Add a dummy m_media entry
|
||||||
|
m_media[fn_compat] = MediaInfo("", "");
|
||||||
|
}
|
||||||
|
|
||||||
|
m_compat_media[fn_compat] = InMemoryMediaInfo(filedata_compat, sha1_base64);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (filedata_to)
|
if (filedata_to)
|
||||||
*filedata_to = std::move(filedata);
|
*filedata_to = std::move(filedata);
|
||||||
return true;
|
return true;
|
||||||
@ -2562,8 +2669,6 @@ void Server::fillMediaCache()
|
|||||||
// The paths are ordered in descending priority
|
// The paths are ordered in descending priority
|
||||||
fs::GetRecursiveDirs(paths, porting::path_user + DIR_DELIM + "textures" + DIR_DELIM + "server");
|
fs::GetRecursiveDirs(paths, porting::path_user + DIR_DELIM + "textures" + DIR_DELIM + "server");
|
||||||
fs::GetRecursiveDirs(paths, m_gamespec.path + DIR_DELIM + "textures");
|
fs::GetRecursiveDirs(paths, m_gamespec.path + DIR_DELIM + "textures");
|
||||||
fs::GetRecursiveDirs(paths, porting::path_share + DIR_DELIM + "builtin" +
|
|
||||||
DIR_DELIM + "game" + DIR_DELIM + "models");
|
|
||||||
m_modmgr->getModsMediaPaths(paths);
|
m_modmgr->getModsMediaPaths(paths);
|
||||||
|
|
||||||
// Collect media file information from paths into cache
|
// Collect media file information from paths into cache
|
||||||
@ -2588,6 +2693,8 @@ void Server::fillMediaCache()
|
|||||||
|
|
||||||
void Server::sendMediaAnnouncement(session_t peer_id, const std::string &lang_code)
|
void Server::sendMediaAnnouncement(session_t peer_id, const std::string &lang_code)
|
||||||
{
|
{
|
||||||
|
const u16 protocol_version = m_clients.getProtocolVersion(peer_id);
|
||||||
|
|
||||||
// Make packet
|
// Make packet
|
||||||
NetworkPacket pkt(TOCLIENT_ANNOUNCE_MEDIA, 0, peer_id);
|
NetworkPacket pkt(TOCLIENT_ANNOUNCE_MEDIA, 0, peer_id);
|
||||||
|
|
||||||
@ -2597,6 +2704,9 @@ void Server::sendMediaAnnouncement(session_t peer_id, const std::string &lang_co
|
|||||||
for (const auto &i : m_media) {
|
for (const auto &i : m_media) {
|
||||||
if (str_ends_with(i.first, ".tr") && !str_ends_with(i.first, lang_suffix))
|
if (str_ends_with(i.first, ".tr") && !str_ends_with(i.first, lang_suffix))
|
||||||
continue;
|
continue;
|
||||||
|
// Skip dummy entries on 5.0+ clients
|
||||||
|
if (protocol_version >= 37 && i.second.sha1_digest.empty())
|
||||||
|
continue;
|
||||||
media_sent++;
|
media_sent++;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2605,7 +2715,18 @@ void Server::sendMediaAnnouncement(session_t peer_id, const std::string &lang_co
|
|||||||
for (const auto &i : m_media) {
|
for (const auto &i : m_media) {
|
||||||
if (str_ends_with(i.first, ".tr") && !str_ends_with(i.first, lang_suffix))
|
if (str_ends_with(i.first, ".tr") && !str_ends_with(i.first, lang_suffix))
|
||||||
continue;
|
continue;
|
||||||
pkt << i.first << i.second.sha1_digest;
|
if (protocol_version >= 37 && i.second.sha1_digest.empty())
|
||||||
|
continue;
|
||||||
|
|
||||||
|
pkt << i.first;
|
||||||
|
|
||||||
|
if (protocol_version < 37 &&
|
||||||
|
m_compat_media.find(i.first) != m_compat_media.end()) {
|
||||||
|
pkt << m_compat_media[i.first].sha1_digest;
|
||||||
|
} else {
|
||||||
|
FATAL_ERROR_IF(i.second.sha1_digest.empty(), "Attempt to send dummy media");
|
||||||
|
pkt << i.second.sha1_digest;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pkt << g_settings->get("remote_media");
|
pkt << g_settings->get("remote_media");
|
||||||
@ -2645,6 +2766,7 @@ void Server::sendRequestedMedia(session_t peer_id,
|
|||||||
|
|
||||||
u32 file_size_bunch_total = 0;
|
u32 file_size_bunch_total = 0;
|
||||||
|
|
||||||
|
const u16 protocol_version = m_clients.getProtocolVersion(peer_id);
|
||||||
for (const std::string &name : tosend) {
|
for (const std::string &name : tosend) {
|
||||||
if (m_media.find(name) == m_media.end()) {
|
if (m_media.find(name) == m_media.end()) {
|
||||||
errorstream<<"Server::sendRequestedMedia(): Client asked for "
|
errorstream<<"Server::sendRequestedMedia(): Client asked for "
|
||||||
@ -2655,6 +2777,20 @@ void Server::sendRequestedMedia(session_t peer_id,
|
|||||||
//TODO get path + name
|
//TODO get path + name
|
||||||
std::string tpath = m_media[name].path;
|
std::string tpath = m_media[name].path;
|
||||||
|
|
||||||
|
// Use compatibility media on older clients
|
||||||
|
if (protocol_version < 37 &&
|
||||||
|
m_compat_media.find(name) != m_compat_media.end()) {
|
||||||
|
file_bunches[file_bunches.size()-1].emplace_back(name, tpath,
|
||||||
|
m_compat_media[name].data);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (tpath.empty()) {
|
||||||
|
errorstream<<"Server::sendRequestedMedia(): New client asked for "
|
||||||
|
<<"compatibility media file \""<<(name)<<"\""<<std::endl;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
// Read data
|
// Read data
|
||||||
std::ifstream fis(tpath.c_str(), std::ios_base::binary);
|
std::ifstream fis(tpath.c_str(), std::ios_base::binary);
|
||||||
if(!fis.good()){
|
if(!fis.good()){
|
||||||
|
25
src/server.h
25
src/server.h
@ -91,6 +91,19 @@ struct MediaInfo
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct InMemoryMediaInfo
|
||||||
|
{
|
||||||
|
std::string data;
|
||||||
|
std::string sha1_digest;
|
||||||
|
|
||||||
|
InMemoryMediaInfo(const std::string &data_="",
|
||||||
|
const std::string &sha1_digest_=""):
|
||||||
|
data(data_),
|
||||||
|
sha1_digest(sha1_digest_)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
struct ServerSoundParams
|
struct ServerSoundParams
|
||||||
{
|
{
|
||||||
enum Type {
|
enum Type {
|
||||||
@ -376,6 +389,15 @@ public:
|
|||||||
// Environment mutex (envlock)
|
// Environment mutex (envlock)
|
||||||
std::mutex m_env_mutex;
|
std::mutex m_env_mutex;
|
||||||
|
|
||||||
|
inline bool isCompatPlayerModel(const std::string &model_name)
|
||||||
|
{
|
||||||
|
return std::find(m_compat_player_models.begin(), m_compat_player_models.end(), model_name) != m_compat_player_models.end();
|
||||||
|
}
|
||||||
|
const std::vector<std::string> getCompatPlayerModels()
|
||||||
|
{
|
||||||
|
return m_compat_player_models;
|
||||||
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
friend class EmergeThread;
|
friend class EmergeThread;
|
||||||
friend class RemoteClient;
|
friend class RemoteClient;
|
||||||
@ -649,6 +671,7 @@ private:
|
|||||||
|
|
||||||
// media files known to server
|
// media files known to server
|
||||||
std::unordered_map<std::string, MediaInfo> m_media;
|
std::unordered_map<std::string, MediaInfo> m_media;
|
||||||
|
std::unordered_map<std::string, InMemoryMediaInfo> m_compat_media;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Sounds
|
Sounds
|
||||||
@ -682,6 +705,8 @@ private:
|
|||||||
MetricCounterPtr m_aom_buffer_counter;
|
MetricCounterPtr m_aom_buffer_counter;
|
||||||
MetricCounterPtr m_packet_recv_counter;
|
MetricCounterPtr m_packet_recv_counter;
|
||||||
MetricCounterPtr m_packet_recv_processed_counter;
|
MetricCounterPtr m_packet_recv_processed_counter;
|
||||||
|
|
||||||
|
std::vector<std::string> m_compat_player_models;
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -592,13 +592,11 @@ std::string PlayerSAO::getPropertyPacket(const u16 protocol_version)
|
|||||||
m_prop.is_visible = (true);
|
m_prop.is_visible = (true);
|
||||||
|
|
||||||
ObjectProperties prop = m_prop;
|
ObjectProperties prop = m_prop;
|
||||||
if (protocol_version < 37 && (m_prop.mesh == "3d_armor_character.b3d" ||
|
|
||||||
m_prop.mesh == "character.b3d" ||
|
// Use the renamed model if compat_send_original_model is enabled
|
||||||
m_prop.mesh == "skinsdb_3d_armor_character_5.b3d")) {
|
if (protocol_version < 37 && m_env->getCompatSendOriginalModel() &&
|
||||||
prop.mesh = "mc_compat_character.b3d";
|
m_env->isCompatPlayerModel(m_prop.mesh)) {
|
||||||
for (u16 i = prop.textures.size(); i < 5; i++) {
|
prop.mesh = "_mc_compat_" + m_prop.mesh;
|
||||||
prop.textures.emplace_back("blank.png");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Remove a one-node offset from a copy of the object properties for MT 0.4
|
// Remove a one-node offset from a copy of the object properties for MT 0.4
|
||||||
|
@ -467,6 +467,9 @@ ServerEnvironment::ServerEnvironment(ServerMap *map,
|
|||||||
|
|
||||||
m_player_database = openPlayerDatabase(player_backend_name, path_world, conf);
|
m_player_database = openPlayerDatabase(player_backend_name, path_world, conf);
|
||||||
m_auth_database = openAuthDatabase(auth_backend_name, path_world, conf);
|
m_auth_database = openAuthDatabase(auth_backend_name, path_world, conf);
|
||||||
|
|
||||||
|
m_compat_send_original_model = !server->getCompatPlayerModels().empty() &&
|
||||||
|
g_settings->getBool("compat_send_original_model");
|
||||||
}
|
}
|
||||||
|
|
||||||
ServerEnvironment::~ServerEnvironment()
|
ServerEnvironment::~ServerEnvironment()
|
||||||
@ -899,7 +902,7 @@ public:
|
|||||||
for (ActiveABM &aabm : *m_aabms[c]) {
|
for (ActiveABM &aabm : *m_aabms[c]) {
|
||||||
if ((p.Y < aabm.min_y) || (p.Y > aabm.max_y))
|
if ((p.Y < aabm.min_y) || (p.Y > aabm.max_y))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
if (myrand() % aabm.chance != 0)
|
if (myrand() % aabm.chance != 0)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
@ -2400,3 +2403,8 @@ bool ServerEnvironment::migrateAuthDatabase(
|
|||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const bool ServerEnvironment::isCompatPlayerModel(const std::string &model_name)
|
||||||
|
{
|
||||||
|
return m_server->isCompatPlayerModel(model_name);
|
||||||
|
}
|
||||||
|
@ -25,6 +25,7 @@ with this program; if not, write to the Free Software Foundation, Inc.,
|
|||||||
#include "settings.h"
|
#include "settings.h"
|
||||||
#include "server/activeobjectmgr.h"
|
#include "server/activeobjectmgr.h"
|
||||||
#include "util/numeric.h"
|
#include "util/numeric.h"
|
||||||
|
#include <algorithm>
|
||||||
#include <set>
|
#include <set>
|
||||||
#include <random>
|
#include <random>
|
||||||
|
|
||||||
@ -364,6 +365,9 @@ public:
|
|||||||
AuthDatabase *getAuthDatabase() { return m_auth_database; }
|
AuthDatabase *getAuthDatabase() { return m_auth_database; }
|
||||||
static bool migrateAuthDatabase(const GameParams &game_params,
|
static bool migrateAuthDatabase(const GameParams &game_params,
|
||||||
const Settings &cmd_args);
|
const Settings &cmd_args);
|
||||||
|
|
||||||
|
const bool isCompatPlayerModel(const std::string &model_name);
|
||||||
|
inline bool getCompatSendOriginalModel() { return m_compat_send_original_model; }
|
||||||
private:
|
private:
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -478,5 +482,8 @@ private:
|
|||||||
std::unordered_map<u32, float> m_particle_spawners;
|
std::unordered_map<u32, float> m_particle_spawners;
|
||||||
std::unordered_map<u32, u16> m_particle_spawner_attachments;
|
std::unordered_map<u32, u16> m_particle_spawner_attachments;
|
||||||
|
|
||||||
|
std::vector<std::string> m_compat_player_models;
|
||||||
|
bool m_compat_send_original_model;
|
||||||
|
|
||||||
ServerActiveObject* createSAO(ActiveObjectType type, v3f pos, const std::string &data);
|
ServerActiveObject* createSAO(ActiveObjectType type, v3f pos, const std::string &data);
|
||||||
};
|
};
|
||||||
|
Loading…
x
Reference in New Issue
Block a user