Files
@ r27835:eabfaa878ced
Branch filter:
Location: cpp/openttd-patchpack/source/src/network/core/game_info.cpp - annotation
r27835:eabfaa878ced
14.1 KiB
text/x-c
Add: calendar date for Survey results
This means no heuristics is possible on around which date people
play the game.
This means no heuristics is possible on around which date people
play the game.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 | r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r27148:4e041ae27b9d r25289:7c44ede418ad r25289:7c44ede418ad r25824:08d7cb74dd5d r25824:08d7cb74dd5d r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25682:1d3bb3fedcce r25289:7c44ede418ad r25688:4e6b21bba5fc r25289:7c44ede418ad r25682:1d3bb3fedcce r25289:7c44ede418ad r25682:1d3bb3fedcce r25777:5dac974697af r25682:1d3bb3fedcce r25682:1d3bb3fedcce r25682:1d3bb3fedcce r25682:1d3bb3fedcce r25682:1d3bb3fedcce r25682:1d3bb3fedcce r25682:1d3bb3fedcce r25682:1d3bb3fedcce r25682:1d3bb3fedcce r25682:1d3bb3fedcce r25682:1d3bb3fedcce r25682:1d3bb3fedcce r25682:1d3bb3fedcce r25682:1d3bb3fedcce r25682:1d3bb3fedcce r25682:1d3bb3fedcce r25289:7c44ede418ad r25682:1d3bb3fedcce r25655:1030dcb7eb52 r25289:7c44ede418ad r25289:7c44ede418ad r25688:4e6b21bba5fc r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25688:4e6b21bba5fc r25289:7c44ede418ad r25289:7c44ede418ad r25688:4e6b21bba5fc r25289:7c44ede418ad r25688:4e6b21bba5fc r25688:4e6b21bba5fc r25688:4e6b21bba5fc r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25688:4e6b21bba5fc r25289:7c44ede418ad r25688:4e6b21bba5fc r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25688:4e6b21bba5fc r25688:4e6b21bba5fc r25688:4e6b21bba5fc r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25292:6323f849890a r25292:6323f849890a r25292:6323f849890a r25292:6323f849890a r25292:6323f849890a r25688:4e6b21bba5fc r25292:6323f849890a r25292:6323f849890a r25292:6323f849890a r25292:6323f849890a r25292:6323f849890a r25292:6323f849890a r25292:6323f849890a r25292:6323f849890a r25292:6323f849890a r25386:b146841c350c r25386:b146841c350c r25289:7c44ede418ad r25386:b146841c350c r25289:7c44ede418ad r25461:f4423b696b95 r27229:662da0bb155f r25385:ad6834a4b5cc r25385:ad6834a4b5cc r26787:a51c38e4aac5 r26787:a51c38e4aac5 r25385:ad6834a4b5cc r25385:ad6834a4b5cc r25385:ad6834a4b5cc r25289:7c44ede418ad r25396:b4291ae45d23 r25396:b4291ae45d23 r25386:b146841c350c r25289:7c44ede418ad r25386:b146841c350c r25386:b146841c350c r25386:b146841c350c r25386:b146841c350c r25386:b146841c350c r25386:b146841c350c r25815:13efa809f0d9 r25815:13efa809f0d9 r25815:13efa809f0d9 r25815:13efa809f0d9 r25815:13efa809f0d9 r25385:ad6834a4b5cc r25385:ad6834a4b5cc r27148:4e041ae27b9d r25385:ad6834a4b5cc r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25837:d00adbe5597a r25837:d00adbe5597a r25289:7c44ede418ad r25837:d00adbe5597a r25289:7c44ede418ad r25289:7c44ede418ad r27391:048886674223 r25289:7c44ede418ad r25837:d00adbe5597a r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25837:d00adbe5597a r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25837:d00adbe5597a r25837:d00adbe5597a r25837:d00adbe5597a r25824:08d7cb74dd5d r25824:08d7cb74dd5d r27737:728d55b97775 r25824:08d7cb74dd5d r25824:08d7cb74dd5d r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25837:d00adbe5597a r25837:d00adbe5597a r25837:d00adbe5597a r25837:d00adbe5597a r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r27788:71137956fe15 r27788:71137956fe15 r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25874:8e2606e1b5e5 r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25385:ad6834a4b5cc r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25838:9f9591576233 r25289:7c44ede418ad r27229:662da0bb155f r25289:7c44ede418ad r25387:bf3342b9f35a r25837:d00adbe5597a r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25387:bf3342b9f35a r25837:d00adbe5597a r25837:d00adbe5597a r25837:d00adbe5597a r25837:d00adbe5597a r25837:d00adbe5597a r25824:08d7cb74dd5d r25824:08d7cb74dd5d r25824:08d7cb74dd5d r25824:08d7cb74dd5d r25824:08d7cb74dd5d r25824:08d7cb74dd5d r25289:7c44ede418ad r26764:765b5f74f0f2 r26764:765b5f74f0f2 r26764:765b5f74f0f2 r27737:728d55b97775 r25289:7c44ede418ad r25289:7c44ede418ad r26764:765b5f74f0f2 r26764:765b5f74f0f2 r25837:d00adbe5597a r25837:d00adbe5597a r25837:d00adbe5597a r25837:d00adbe5597a r25837:d00adbe5597a r25837:d00adbe5597a r25837:d00adbe5597a r25837:d00adbe5597a r25837:d00adbe5597a r25837:d00adbe5597a r25838:9f9591576233 r25838:9f9591576233 r25838:9f9591576233 r25838:9f9591576233 r25838:9f9591576233 r25838:9f9591576233 r25838:9f9591576233 r25838:9f9591576233 r25837:d00adbe5597a r25837:d00adbe5597a r25837:d00adbe5597a r25837:d00adbe5597a r25289:7c44ede418ad r25837:d00adbe5597a r25837:d00adbe5597a r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r27788:71137956fe15 r27788:71137956fe15 r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25874:8e2606e1b5e5 r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25396:b4291ae45d23 r25396:b4291ae45d23 r25837:d00adbe5597a r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25387:bf3342b9f35a r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25837:d00adbe5597a r25289:7c44ede418ad r25289:7c44ede418ad r25385:ad6834a4b5cc r25289:7c44ede418ad r25289:7c44ede418ad r25854:8c8bc8b61b57 r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r27391:048886674223 r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r27391:048886674223 r25289:7c44ede418ad r25289:7c44ede418ad r25289:7c44ede418ad r25837:d00adbe5597a r25837:d00adbe5597a r25837:d00adbe5597a r25837:d00adbe5597a r25837:d00adbe5597a r25837:d00adbe5597a r25837:d00adbe5597a r25837:d00adbe5597a r25837:d00adbe5597a r25837:d00adbe5597a r25837:d00adbe5597a | /*
* This file is part of OpenTTD.
* OpenTTD 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, version 2.
* OpenTTD 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 OpenTTD. If not, see <http://www.gnu.org/licenses/>.
*/
/**
* @file game_info.cpp Functions to convert NetworkGameInfo to Packet and back.
*/
#include "../../stdafx.h"
#include "game_info.h"
#include "../../core/bitmath_func.hpp"
#include "../../company_base.h"
#include "../../timer/timer_game_calendar.h"
#include "../../debug.h"
#include "../../map_func.h"
#include "../../game/game.hpp"
#include "../../game/game_info.hpp"
#include "../../settings_type.h"
#include "../../string_func.h"
#include "../../rev.h"
#include "../network_func.h"
#include "../network.h"
#include "packet.h"
#include "../../safeguards.h"
/**
* How many hex digits of the git hash to include in network revision string.
* Determined as 10 hex digits + 2 characters for -g/-u/-m prefix.
*/
static const uint GITHASH_SUFFIX_LEN = 12;
NetworkServerGameInfo _network_game_info; ///< Information about our game.
/**
* Get the network version string used by this build.
* The returned string is guaranteed to be at most NETWORK_REVISON_LENGTH bytes including '\0' terminator.
*/
std::string_view GetNetworkRevisionString()
{
static std::string network_revision;
if (network_revision.empty()) {
network_revision = _openttd_revision;
if (_openttd_revision_tagged) {
/* Tagged; do not mangle further, though ensure it's not too long. */
if (network_revision.size() >= NETWORK_REVISION_LENGTH) network_revision.resize(NETWORK_REVISION_LENGTH - 1);
} else {
/* Not tagged; add the githash suffix while ensuring the string does not become too long. */
assert(_openttd_revision_modified < 3);
std::string githash_suffix = fmt::format("-{}{}", "gum"[_openttd_revision_modified], _openttd_revision_hash);
if (githash_suffix.size() > GITHASH_SUFFIX_LEN) githash_suffix.resize(GITHASH_SUFFIX_LEN);
/* Where did the hash start in the original string? Overwrite from that position, unless that would create a too long string. */
size_t hash_end = network_revision.find_last_of('-');
if (hash_end == std::string::npos) hash_end = network_revision.size();
if (hash_end + githash_suffix.size() >= NETWORK_REVISION_LENGTH) hash_end = NETWORK_REVISION_LENGTH - githash_suffix.size() - 1;
/* Replace the git hash in revision string. */
network_revision.replace(hash_end, std::string::npos, githash_suffix);
}
assert(network_revision.size() < NETWORK_REVISION_LENGTH); // size does not include terminator, constant does, hence strictly less than
Debug(net, 3, "Network revision name: {}", network_revision);
}
return network_revision;
}
/**
* Extract the git hash from the revision string.
* @param revision_string The revision string (formatted as DATE-BRANCH-GITHASH).
* @return The git has part of the revision.
*/
static std::string_view ExtractNetworkRevisionHash(std::string_view revision_string)
{
size_t index = revision_string.find_last_of('-');
if (index == std::string::npos) return {};
return revision_string.substr(index);
}
/**
* Checks whether the given version string is compatible with our version.
* First tries to match the full string, if that fails, attempts to compare just git hashes.
* @param other the version string to compare to
*/
bool IsNetworkCompatibleVersion(std::string_view other)
{
if (GetNetworkRevisionString() == other) return true;
/* If this version is tagged, then the revision string must be a complete match,
* since there is no git hash suffix in it.
* This is needed to avoid situations like "1.9.0-beta1" comparing equal to "2.0.0-beta1". */
if (_openttd_revision_tagged) return false;
std::string_view hash1 = ExtractNetworkRevisionHash(GetNetworkRevisionString());
std::string_view hash2 = ExtractNetworkRevisionHash(other);
return hash1 == hash2;
}
/**
* Check if an game entry is compatible with our client.
*/
void CheckGameCompatibility(NetworkGameInfo &ngi)
{
/* Check if we are allowed on this server based on the revision-check. */
ngi.version_compatible = IsNetworkCompatibleVersion(ngi.server_revision);
ngi.compatible = ngi.version_compatible;
/* Check if we have all the GRFs on the client-system too. */
for (const GRFConfig *c = ngi.grfconfig; c != nullptr; c = c->next) {
if (c->status == GCS_NOT_FOUND) ngi.compatible = false;
}
}
/**
* Fill a NetworkServerGameInfo structure with the static content, or things
* that are so static they can be updated on request from a settings change.
*/
void FillStaticNetworkServerGameInfo()
{
_network_game_info.use_password = !_settings_client.network.server_password.empty();
_network_game_info.start_date = TimerGameCalendar::ConvertYMDToDate(_settings_game.game_creation.starting_year, 0, 1);
_network_game_info.clients_max = _settings_client.network.max_clients;
_network_game_info.companies_max = _settings_client.network.max_companies;
_network_game_info.map_width = Map::SizeX();
_network_game_info.map_height = Map::SizeY();
_network_game_info.landscape = _settings_game.game_creation.landscape;
_network_game_info.dedicated = _network_dedicated;
_network_game_info.grfconfig = _grfconfig;
_network_game_info.server_name = _settings_client.network.server_name;
_network_game_info.server_revision = GetNetworkRevisionString();
}
/**
* Get the NetworkServerGameInfo structure with the latest information of the server.
* @return The current NetworkServerGameInfo.
*/
const NetworkServerGameInfo *GetCurrentNetworkServerGameInfo()
{
/* These variables are updated inside _network_game_info as if they are global variables:
* - clients_on
* - invite_code
* These don't need to be updated manually here.
*/
_network_game_info.companies_on = (byte)Company::GetNumItems();
_network_game_info.spectators_on = NetworkSpectatorCount();
_network_game_info.game_date = TimerGameCalendar::date;
return &_network_game_info;
}
/**
* Function that is called for every GRFConfig that is read when receiving
* a NetworkGameInfo. Only grfid and md5sum are set, the rest is zero. This
* function must set all appropriate fields. This GRF is later appended to
* the grfconfig list of the NetworkGameInfo.
* @param config The GRF to handle.
* @param name The name of the NewGRF, empty when unknown.
*/
static void HandleIncomingNetworkGameInfoGRFConfig(GRFConfig *config, std::string name)
{
/* Find the matching GRF file */
const GRFConfig *f = FindGRFConfig(config->ident.grfid, FGCM_EXACT, &config->ident.md5sum);
if (f == nullptr) {
AddGRFTextToList(config->name, name.empty() ? GetString(STR_CONFIG_ERROR_INVALID_GRF_UNKNOWN) : name);
config->status = GCS_NOT_FOUND;
} else {
config->filename = f->filename;
config->name = f->name;
config->info = f->info;
config->url = f->url;
}
SetBit(config->flags, GCF_COPY);
}
/**
* Serializes the NetworkGameInfo struct to the packet.
* @param p the packet to write the data to.
* @param info the NetworkGameInfo struct to serialize from.
*/
void SerializeNetworkGameInfo(Packet *p, const NetworkServerGameInfo *info, bool send_newgrf_names)
{
p->Send_uint8 (NETWORK_GAME_INFO_VERSION);
/*
* Please observe the order.
* The parts must be read in the same order as they are sent!
*/
/* Update the documentation in game_info.h on changes
* to the NetworkGameInfo wire-protocol! */
/* NETWORK_GAME_INFO_VERSION = 6 */
p->Send_uint8(send_newgrf_names ? NST_GRFID_MD5_NAME : NST_GRFID_MD5);
/* NETWORK_GAME_INFO_VERSION = 5 */
GameInfo *game_info = Game::GetInfo();
p->Send_uint32(game_info == nullptr ? -1 : (uint32_t)game_info->GetVersion());
p->Send_string(game_info == nullptr ? "" : game_info->GetName());
/* NETWORK_GAME_INFO_VERSION = 4 */
{
/* Only send the GRF Identification (GRF_ID and MD5 checksum) of
* the GRFs that are needed, i.e. the ones that the server has
* selected in the NewGRF GUI and not the ones that are used due
* to the fact that they are in [newgrf-static] in openttd.cfg */
const GRFConfig *c;
uint count = 0;
/* Count number of GRFs to send information about */
for (c = info->grfconfig; c != nullptr; c = c->next) {
if (!HasBit(c->flags, GCF_STATIC)) count++;
}
p->Send_uint8 (count); // Send number of GRFs
/* Send actual GRF Identifications */
for (c = info->grfconfig; c != nullptr; c = c->next) {
if (HasBit(c->flags, GCF_STATIC)) continue;
SerializeGRFIdentifier(p, &c->ident);
if (send_newgrf_names) p->Send_string(c->GetName());
}
}
/* NETWORK_GAME_INFO_VERSION = 3 */
p->Send_uint32(static_cast<int32_t>(info->game_date));
p->Send_uint32(static_cast<int32_t>(info->start_date));
/* NETWORK_GAME_INFO_VERSION = 2 */
p->Send_uint8 (info->companies_max);
p->Send_uint8 (info->companies_on);
p->Send_uint8 (info->clients_max); // Used to be max-spectators
/* NETWORK_GAME_INFO_VERSION = 1 */
p->Send_string(info->server_name);
p->Send_string(info->server_revision);
p->Send_bool (info->use_password);
p->Send_uint8 (info->clients_max);
p->Send_uint8 (info->clients_on);
p->Send_uint8 (info->spectators_on);
p->Send_uint16(info->map_width);
p->Send_uint16(info->map_height);
p->Send_uint8 (info->landscape);
p->Send_bool (info->dedicated);
}
/**
* Deserializes the NetworkGameInfo struct from the packet.
* @param p the packet to read the data from.
* @param info the NetworkGameInfo to deserialize into.
*/
void DeserializeNetworkGameInfo(Packet *p, NetworkGameInfo *info, const GameInfoNewGRFLookupTable *newgrf_lookup_table)
{
static const TimerGameCalendar::Date MAX_DATE = TimerGameCalendar::ConvertYMDToDate(MAX_YEAR, 11, 31); // December is month 11
byte game_info_version = p->Recv_uint8();
NewGRFSerializationType newgrf_serialisation = NST_GRFID_MD5;
/*
* Please observe the order.
* The parts must be read in the same order as they are sent!
*/
/* Update the documentation in game_info.h on changes
* to the NetworkGameInfo wire-protocol! */
switch (game_info_version) {
case 6:
newgrf_serialisation = (NewGRFSerializationType)p->Recv_uint8();
if (newgrf_serialisation >= NST_END) return;
FALLTHROUGH;
case 5: {
info->gamescript_version = (int)p->Recv_uint32();
info->gamescript_name = p->Recv_string(NETWORK_NAME_LENGTH);
FALLTHROUGH;
}
case 4: {
/* Ensure that the maximum number of NewGRFs and the field in the network
* protocol are matched to eachother. If that is not the case anymore a
* check must be added to ensure the received data is still valid. */
static_assert(std::numeric_limits<uint8_t>::max() == NETWORK_MAX_GRF_COUNT);
uint num_grfs = p->Recv_uint8();
GRFConfig **dst = &info->grfconfig;
for (uint i = 0; i < num_grfs; i++) {
NamedGRFIdentifier grf;
switch (newgrf_serialisation) {
case NST_GRFID_MD5:
DeserializeGRFIdentifier(p, &grf.ident);
break;
case NST_GRFID_MD5_NAME:
DeserializeGRFIdentifierWithName(p, &grf);
break;
case NST_LOOKUP_ID: {
if (newgrf_lookup_table == nullptr) return;
auto it = newgrf_lookup_table->find(p->Recv_uint32());
if (it == newgrf_lookup_table->end()) return;
grf = it->second;
break;
}
default:
NOT_REACHED();
}
GRFConfig *c = new GRFConfig();
c->ident = grf.ident;
HandleIncomingNetworkGameInfoGRFConfig(c, grf.name);
/* Append GRFConfig to the list */
*dst = c;
dst = &c->next;
}
FALLTHROUGH;
}
case 3:
info->game_date = Clamp(p->Recv_uint32(), 0, static_cast<int32_t>(MAX_DATE));
info->start_date = Clamp(p->Recv_uint32(), 0, static_cast<int32_t>(MAX_DATE));
FALLTHROUGH;
case 2:
info->companies_max = p->Recv_uint8 ();
info->companies_on = p->Recv_uint8 ();
p->Recv_uint8(); // Used to contain max-spectators.
FALLTHROUGH;
case 1:
info->server_name = p->Recv_string(NETWORK_NAME_LENGTH);
info->server_revision = p->Recv_string(NETWORK_REVISION_LENGTH);
if (game_info_version < 6) p->Recv_uint8 (); // Used to contain server-lang.
info->use_password = p->Recv_bool ();
info->clients_max = p->Recv_uint8 ();
info->clients_on = p->Recv_uint8 ();
info->spectators_on = p->Recv_uint8 ();
if (game_info_version < 3) { // 16 bits dates got scrapped and are read earlier
info->game_date = p->Recv_uint16() + DAYS_TILL_ORIGINAL_BASE_YEAR;
info->start_date = p->Recv_uint16() + DAYS_TILL_ORIGINAL_BASE_YEAR;
}
if (game_info_version < 6) while (p->Recv_uint8() != 0) {} // Used to contain the map-name.
info->map_width = p->Recv_uint16();
info->map_height = p->Recv_uint16();
info->landscape = p->Recv_uint8 ();
info->dedicated = p->Recv_bool ();
if (info->landscape >= NUM_LANDSCAPE) info->landscape = 0;
}
}
/**
* Serializes the GRFIdentifier (GRF ID and MD5 checksum) to the packet
* @param p the packet to write the data to.
* @param grf the GRFIdentifier to serialize.
*/
void SerializeGRFIdentifier(Packet *p, const GRFIdentifier *grf)
{
p->Send_uint32(grf->grfid);
for (size_t j = 0; j < grf->md5sum.size(); j++) {
p->Send_uint8(grf->md5sum[j]);
}
}
/**
* Deserializes the GRFIdentifier (GRF ID and MD5 checksum) from the packet
* @param p the packet to read the data from.
* @param grf the GRFIdentifier to deserialize.
*/
void DeserializeGRFIdentifier(Packet *p, GRFIdentifier *grf)
{
grf->grfid = p->Recv_uint32();
for (size_t j = 0; j < grf->md5sum.size(); j++) {
grf->md5sum[j] = p->Recv_uint8();
}
}
/**
* Deserializes the NamedGRFIdentifier (GRF ID, MD5 checksum and name) from the packet
* @param p the packet to read the data from.
* @param grf the NamedGRFIdentifier to deserialize.
*/
void DeserializeGRFIdentifierWithName(Packet *p, NamedGRFIdentifier *grf)
{
DeserializeGRFIdentifier(p, &grf->ident);
grf->name = p->Recv_string(NETWORK_GRF_NAME_LENGTH);
}
|