mirror of
https://github.com/JGRennison/OpenTTD-patches.git
synced 2024-11-17 21:25:40 +00:00
576 lines
16 KiB
C++
576 lines
16 KiB
C++
/* $Id$ */
|
|
|
|
/** @file newgrf_config.cpp */
|
|
|
|
#include "stdafx.h"
|
|
#include "openttd.h"
|
|
#include "debug.h"
|
|
#include "variables.h"
|
|
#include "string.h"
|
|
#include "saveload.h"
|
|
#include "md5.h"
|
|
#include "network/network_data.h"
|
|
#include "newgrf.h"
|
|
#include "newgrf_config.h"
|
|
|
|
#include "fileio.h"
|
|
#include "fios.h"
|
|
#include <sys/stat.h>
|
|
|
|
#ifdef WIN32
|
|
# include <io.h>
|
|
#endif /* WIN32 */
|
|
|
|
|
|
GRFConfig *_all_grfs;
|
|
GRFConfig *_grfconfig;
|
|
GRFConfig *_grfconfig_newgame;
|
|
GRFConfig *_grfconfig_static;
|
|
|
|
|
|
/* Calculate the MD5 Sum for a GRF */
|
|
static bool CalcGRFMD5Sum(GRFConfig *config)
|
|
{
|
|
FILE *f;
|
|
md5_state_t md5state;
|
|
md5_byte_t buffer[1024];
|
|
size_t len, size;
|
|
|
|
/* open the file */
|
|
f = FioFOpenFile(config->filename, "rb", DATA_DIR, &size);
|
|
if (f == NULL) return false;
|
|
|
|
/* calculate md5sum */
|
|
md5_init(&md5state);
|
|
while ((len = fread(buffer, 1, (size > sizeof(buffer)) ? sizeof(buffer) : size, f)) != 0 && size != 0) {
|
|
size -= len;
|
|
md5_append(&md5state, buffer, len);
|
|
}
|
|
md5_finish(&md5state, config->md5sum);
|
|
|
|
FioFCloseFile(f);
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
/* Find the GRFID and calculate the md5sum */
|
|
bool FillGRFDetails(GRFConfig *config, bool is_static)
|
|
{
|
|
if (!FioCheckFileExists(config->filename)) {
|
|
config->status = GCS_NOT_FOUND;
|
|
return false;
|
|
}
|
|
|
|
/* Find and load the Action 8 information */
|
|
LoadNewGRFFile(config, CONFIG_SLOT, GLS_FILESCAN);
|
|
|
|
/* Skip if the grfid is 0 (not read) or 0xFFFFFFFF (ttdp system grf) */
|
|
if (config->grfid == 0 || config->grfid == 0xFFFFFFFF || config->IsOpenTTDBaseGRF()) return false;
|
|
|
|
if (is_static) {
|
|
/* Perform a 'safety scan' for static GRFs */
|
|
LoadNewGRFFile(config, 62, GLS_SAFETYSCAN);
|
|
|
|
/* GCF_UNSAFE is set if GLS_SAFETYSCAN finds unsafe actions */
|
|
if (HasBit(config->flags, GCF_UNSAFE)) return false;
|
|
}
|
|
|
|
return CalcGRFMD5Sum(config);
|
|
}
|
|
|
|
|
|
void ClearGRFConfig(GRFConfig **config)
|
|
{
|
|
/* GCF_COPY as in NOT strdupped/alloced the filename, name and info */
|
|
if (!HasBit((*config)->flags, GCF_COPY)) {
|
|
free((*config)->filename);
|
|
free((*config)->name);
|
|
free((*config)->info);
|
|
|
|
if ((*config)->error != NULL) {
|
|
free((*config)->error->custom_message);
|
|
free((*config)->error->data);
|
|
free((*config)->error);
|
|
}
|
|
}
|
|
free(*config);
|
|
*config = NULL;
|
|
}
|
|
|
|
|
|
/* Clear a GRF Config list */
|
|
void ClearGRFConfigList(GRFConfig **config)
|
|
{
|
|
GRFConfig *c, *next;
|
|
for (c = *config; c != NULL; c = next) {
|
|
next = c->next;
|
|
ClearGRFConfig(&c);
|
|
}
|
|
*config = NULL;
|
|
}
|
|
|
|
|
|
/** Copy a GRF Config list
|
|
* @param dst pointer to destination list
|
|
* @param src pointer to source list values
|
|
* @param init_only the copied GRF will be processed up to GLS_INIT
|
|
* @return pointer to the last value added to the destination list */
|
|
GRFConfig **CopyGRFConfigList(GRFConfig **dst, const GRFConfig *src, bool init_only)
|
|
{
|
|
/* Clear destination as it will be overwritten */
|
|
ClearGRFConfigList(dst);
|
|
for (; src != NULL; src = src->next) {
|
|
GRFConfig *c = CallocT<GRFConfig>(1);
|
|
*c = *src;
|
|
if (src->filename != NULL) c->filename = strdup(src->filename);
|
|
if (src->name != NULL) c->name = strdup(src->name);
|
|
if (src->info != NULL) c->info = strdup(src->info);
|
|
if (src->error != NULL) {
|
|
c->error = CallocT<GRFError>(1);
|
|
memcpy(c->error, src->error, sizeof(GRFError));
|
|
if (src->error->data != NULL) c->error->data = strdup(src->error->data);
|
|
if (src->error->custom_message != NULL) c->error->custom_message = strdup(src->error->custom_message);
|
|
}
|
|
|
|
ClrBit(c->flags, GCF_INIT_ONLY);
|
|
if (init_only) SetBit(c->flags, GCF_INIT_ONLY);
|
|
|
|
*dst = c;
|
|
dst = &c->next;
|
|
}
|
|
|
|
return dst;
|
|
}
|
|
|
|
/**
|
|
* Removes duplicates from lists of GRFConfigs. These duplicates
|
|
* are introduced when the _grfconfig_static GRFs are appended
|
|
* to the _grfconfig on a newgame or savegame. As the parameters
|
|
* of the static GRFs could be different that the parameters of
|
|
* the ones used non-statically. This can result in desyncs in
|
|
* multiplayers, so the duplicate static GRFs have to be removed.
|
|
*
|
|
* This function _assumes_ that all static GRFs are placed after
|
|
* the non-static GRFs.
|
|
*
|
|
* @param list the list to remove the duplicates from
|
|
*/
|
|
static void RemoveDuplicatesFromGRFConfigList(GRFConfig *list)
|
|
{
|
|
GRFConfig *prev;
|
|
GRFConfig *cur;
|
|
|
|
if (list == NULL) return;
|
|
|
|
for (prev = list, cur = list->next; cur != NULL; prev = cur, cur = cur->next) {
|
|
if (cur->grfid != list->grfid) continue;
|
|
|
|
prev->next = cur->next;
|
|
ClearGRFConfig(&cur);
|
|
cur = prev; // Just go back one so it continues as normal later on
|
|
}
|
|
|
|
RemoveDuplicatesFromGRFConfigList(list->next);
|
|
}
|
|
|
|
/**
|
|
* Appends the static GRFs to a list of GRFs
|
|
* @param dst the head of the list to add to
|
|
*/
|
|
void AppendStaticGRFConfigs(GRFConfig **dst)
|
|
{
|
|
GRFConfig **tail = dst;
|
|
while (*tail != NULL) tail = &(*tail)->next;
|
|
|
|
CopyGRFConfigList(tail, _grfconfig_static, false);
|
|
RemoveDuplicatesFromGRFConfigList(*dst);
|
|
}
|
|
|
|
/** Appends an element to a list of GRFs
|
|
* @param dst the head of the list to add to
|
|
* @param el the new tail to be */
|
|
void AppendToGRFConfigList(GRFConfig **dst, GRFConfig *el)
|
|
{
|
|
GRFConfig **tail = dst;
|
|
while (*tail != NULL) tail = &(*tail)->next;
|
|
*tail = el;
|
|
|
|
RemoveDuplicatesFromGRFConfigList(*dst);
|
|
}
|
|
|
|
|
|
/* Reset the current GRF Config to either blank or newgame settings */
|
|
void ResetGRFConfig(bool defaults)
|
|
{
|
|
CopyGRFConfigList(&_grfconfig, _grfconfig_newgame, !defaults);
|
|
AppendStaticGRFConfigs(&_grfconfig);
|
|
}
|
|
|
|
|
|
/** Check if all GRFs in the GRF config from a savegame can be loaded.
|
|
* @return will return any of the following 3 values:<br>
|
|
* <ul>
|
|
* <li> GLC_ALL_GOOD: No problems occured, all GRF files were found and loaded
|
|
* <li> GLC_COMPATIBLE: For one or more GRF's no exact match was found, but a
|
|
* compatible GRF with the same grfid was found and used instead
|
|
* <li> GLC_NOT_FOUND: For one or more GRF's no match was found at all
|
|
* </ul> */
|
|
GRFListCompatibility IsGoodGRFConfigList()
|
|
{
|
|
GRFListCompatibility res = GLC_ALL_GOOD;
|
|
|
|
for (GRFConfig *c = _grfconfig; c != NULL; c = c->next) {
|
|
const GRFConfig *f = FindGRFConfig(c->grfid, c->md5sum);
|
|
if (f == NULL) {
|
|
char buf[256];
|
|
|
|
/* If we have not found the exactly matching GRF try to find one with the
|
|
* same grfid, as it most likely is compatible */
|
|
f = FindGRFConfig(c->grfid);
|
|
if (f != NULL) {
|
|
md5sumToString(buf, lastof(buf), c->md5sum);
|
|
DEBUG(grf, 1, "NewGRF %08X (%s) not found; checksum %s. Compatibility mode on", BSWAP32(c->grfid), c->filename, buf);
|
|
SetBit(c->flags, GCF_COMPATIBLE);
|
|
|
|
/* Non-found has precedence over compatibility load */
|
|
if (res != GLC_NOT_FOUND) res = GLC_COMPATIBLE;
|
|
goto compatible_grf;
|
|
}
|
|
|
|
/* No compatible grf was found, mark it as disabled */
|
|
md5sumToString(buf, lastof(buf), c->md5sum);
|
|
DEBUG(grf, 0, "NewGRF %08X (%s) not found; checksum %s", BSWAP32(c->grfid), c->filename, buf);
|
|
|
|
c->status = GCS_NOT_FOUND;
|
|
res = GLC_NOT_FOUND;
|
|
} else {
|
|
compatible_grf:
|
|
DEBUG(grf, 1, "Loading GRF %08X from %s", BSWAP32(f->grfid), f->filename);
|
|
/* The filename could be the filename as in the savegame. As we need
|
|
* to load the GRF here, we need the correct filename, so overwrite that
|
|
* in any case and set the name and info when it is not set already.
|
|
* When the GCF_COPY flag is set, it is certain that the filename is
|
|
* already a local one, so there is no need to replace it. */
|
|
if (!HasBit(c->flags, GCF_COPY)) {
|
|
free(c->filename);
|
|
c->filename = strdup(f->filename);
|
|
memcpy(c->md5sum, f->md5sum, sizeof(c->md5sum));
|
|
if (c->name == NULL && f->name != NULL) c->name = strdup(f->name);
|
|
if (c->info == NULL && f->info != NULL) c->info = strdup(f->info);
|
|
c->error = NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
static bool ScanPathAddGrf(const char *filename)
|
|
{
|
|
GRFConfig *c = CallocT<GRFConfig>(1);
|
|
c->filename = strdup(filename);
|
|
|
|
bool added = true;
|
|
if (FillGRFDetails(c, false)) {
|
|
if (_all_grfs == NULL) {
|
|
_all_grfs = c;
|
|
} else {
|
|
/* Insert file into list at a position determined by its
|
|
* name, so the list is sorted as we go along */
|
|
GRFConfig **pd, *d;
|
|
bool stop = false;
|
|
for (pd = &_all_grfs; (d = *pd) != NULL; pd = &d->next) {
|
|
if (c->grfid == d->grfid && memcmp(c->md5sum, d->md5sum, sizeof(c->md5sum)) == 0) added = false;
|
|
/* Because there can be multiple grfs with the same name, make sure we checked all grfs with the same name,
|
|
* before inserting the entry. So insert a new grf at the end of all grfs with the same name, instead of
|
|
* just after the first with the same name. Avoids doubles in the list. */
|
|
if (strcasecmp(c->name, d->name) <= 0) stop = true;
|
|
else if (stop) break;
|
|
}
|
|
if (added) {
|
|
c->next = d;
|
|
*pd = c;
|
|
}
|
|
}
|
|
} else {
|
|
added = false;
|
|
}
|
|
|
|
if (!added) {
|
|
/* File couldn't be opened, or is either not a NewGRF or is a
|
|
* 'system' NewGRF or it's already known, so forget about it. */
|
|
free(c->filename);
|
|
free(c->name);
|
|
free(c->info);
|
|
free(c);
|
|
}
|
|
|
|
return added;
|
|
}
|
|
|
|
/* Scan a path for NewGRFs */
|
|
static uint ScanPath(const char *path, int basepath_length)
|
|
{
|
|
extern bool FiosIsValidFile(const char *path, const struct dirent *ent, struct stat *sb);
|
|
|
|
uint num = 0;
|
|
struct stat sb;
|
|
struct dirent *dirent;
|
|
DIR *dir;
|
|
|
|
if (path == NULL || (dir = ttd_opendir(path)) == NULL) return 0;
|
|
|
|
while ((dirent = readdir(dir)) != NULL) {
|
|
const char *d_name = FS2OTTD(dirent->d_name);
|
|
char filename[MAX_PATH];
|
|
|
|
if (!FiosIsValidFile(path, dirent, &sb)) continue;
|
|
|
|
snprintf(filename, lengthof(filename), "%s%s", path, d_name);
|
|
|
|
if (sb.st_mode & S_IFDIR) {
|
|
/* Directory */
|
|
if (strcmp(d_name, ".") == 0 || strcmp(d_name, "..") == 0) continue;
|
|
AppendPathSeparator(filename, lengthof(filename));
|
|
num += ScanPath(filename, basepath_length);
|
|
} else if (sb.st_mode & S_IFREG) {
|
|
/* File */
|
|
char *ext = strrchr(filename, '.');
|
|
|
|
/* If no extension or extension isn't .grf, skip the file */
|
|
if (ext == NULL) continue;
|
|
if (strcasecmp(ext, ".grf") != 0) continue;
|
|
|
|
if (ScanPathAddGrf(filename + basepath_length)) num++;
|
|
}
|
|
}
|
|
|
|
closedir(dir);
|
|
|
|
return num;
|
|
}
|
|
|
|
static uint ScanTar(TarFileList::iterator tar)
|
|
{
|
|
uint num = 0;
|
|
const char *filename = (*tar).first.c_str();
|
|
const char *ext = strrchr(filename, '.');
|
|
|
|
/* If no extension or extension isn't .grf, skip the file */
|
|
if (ext == NULL) return false;
|
|
if (strcasecmp(ext, ".grf") != 0) return false;
|
|
|
|
if (ScanPathAddGrf(filename)) num++;
|
|
|
|
return num;
|
|
}
|
|
|
|
/**
|
|
* Simple sorter for GRFS
|
|
* @param p1 the first GRFConfig *
|
|
* @param p2 the second GRFConfig *
|
|
* @return the same strcmp would return for the name of the NewGRF.
|
|
*/
|
|
static int CDECL GRFSorter(const void *p1, const void *p2)
|
|
{
|
|
const GRFConfig *c1 = *(const GRFConfig **)p1;
|
|
const GRFConfig *c2 = *(const GRFConfig **)p2;
|
|
|
|
return strcmp(c1->name != NULL ? c1->name : c1->filename,
|
|
c2->name != NULL ? c2->name : c2->filename);
|
|
}
|
|
|
|
/* Scan for all NewGRFs */
|
|
void ScanNewGRFFiles()
|
|
{
|
|
Searchpath sp;
|
|
char path[MAX_PATH];
|
|
TarFileList::iterator tar;
|
|
uint num = 0;
|
|
|
|
ClearGRFConfigList(&_all_grfs);
|
|
|
|
DEBUG(grf, 1, "Scanning for NewGRFs");
|
|
FOR_ALL_SEARCHPATHS(sp) {
|
|
FioAppendDirectory(path, MAX_PATH, sp, DATA_DIR);
|
|
num += ScanPath(path, strlen(path));
|
|
}
|
|
FOR_ALL_TARS(tar) {
|
|
num += ScanTar(tar);
|
|
}
|
|
|
|
DEBUG(grf, 1, "Scan complete, found %d files", num);
|
|
if (num == 0 || _all_grfs == NULL) return;
|
|
|
|
/* Sort the linked list using quicksort.
|
|
* For that we first have to make an array, the qsort and
|
|
* then remake the linked list. */
|
|
GRFConfig **to_sort = MallocT<GRFConfig*>(num);
|
|
if (to_sort == NULL) return; // No memory, then don't sort
|
|
|
|
uint i = 0;
|
|
for (GRFConfig *p = _all_grfs; p != NULL; p = p->next, i++) {
|
|
to_sort[i] = p;
|
|
}
|
|
/* Number of files is not necessarily right */
|
|
num = i;
|
|
|
|
qsort(to_sort, num, sizeof(GRFConfig*), GRFSorter);
|
|
|
|
for (i = 1; i < num; i++) {
|
|
to_sort[i - 1]->next = to_sort[i];
|
|
}
|
|
to_sort[num - 1]->next = NULL;
|
|
_all_grfs = to_sort[0];
|
|
}
|
|
|
|
|
|
/* Find a NewGRF in the scanned list, if md5sum is NULL, we don't care about it*/
|
|
const GRFConfig *FindGRFConfig(uint32 grfid, const uint8 *md5sum)
|
|
{
|
|
for (const GRFConfig *c = _all_grfs; c != NULL; c = c->next) {
|
|
if (c->grfid == grfid) {
|
|
if (md5sum == NULL) return c;
|
|
|
|
if (memcmp(md5sum, c->md5sum, sizeof(c->md5sum)) == 0) return c;
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
#ifdef ENABLE_NETWORK
|
|
|
|
/** Structure for UnknownGRFs; this is a lightweight variant of GRFConfig */
|
|
struct UnknownGRF : public GRFIdentifier {
|
|
UnknownGRF *next;
|
|
char name[NETWORK_GRF_NAME_LENGTH];
|
|
};
|
|
|
|
/**
|
|
* Finds the name of a NewGRF in the list of names for unknown GRFs. An
|
|
* unknown GRF is a GRF where the .grf is not found during scanning.
|
|
*
|
|
* The names are resolved via UDP calls to servers that should know the name,
|
|
* though the replies may not come. This leaves "<Unknown>" as name, though
|
|
* that shouldn't matter _very_ much as they need GRF crawler or so to look
|
|
* up the GRF anyway and that works better with the GRF ID.
|
|
*
|
|
* @param grfid the GRF ID part of the 'unique' GRF identifier
|
|
* @param md5sum the MD5 checksum part of the 'unique' GRF identifier
|
|
* @param create whether to create a new GRFConfig if the GRFConfig did not
|
|
* exist in the fake list of GRFConfigs.
|
|
* @return the GRFConfig with the given GRF ID and MD5 checksum or NULL when
|
|
* it does not exist and create is false. This value must NEVER be
|
|
* freed by the caller.
|
|
*/
|
|
char *FindUnknownGRFName(uint32 grfid, uint8 *md5sum, bool create)
|
|
{
|
|
UnknownGRF *grf;
|
|
static UnknownGRF *unknown_grfs = NULL;
|
|
|
|
for (grf = unknown_grfs; grf != NULL; grf = grf->next) {
|
|
if (grf->grfid == grfid) {
|
|
if (memcmp(md5sum, grf->md5sum, sizeof(grf->md5sum)) == 0) return grf->name;
|
|
}
|
|
}
|
|
|
|
if (!create) return NULL;
|
|
|
|
grf = CallocT<UnknownGRF>(1);
|
|
grf->grfid = grfid;
|
|
grf->next = unknown_grfs;
|
|
ttd_strlcpy(grf->name, UNKNOWN_GRF_NAME_PLACEHOLDER, sizeof(grf->name));
|
|
memcpy(grf->md5sum, md5sum, sizeof(grf->md5sum));
|
|
|
|
unknown_grfs = grf;
|
|
return grf->name;
|
|
}
|
|
|
|
#endif /* ENABLE_NETWORK */
|
|
|
|
|
|
/* Retrieve a NewGRF from the current config by its grfid */
|
|
GRFConfig *GetGRFConfig(uint32 grfid)
|
|
{
|
|
GRFConfig *c;
|
|
|
|
for (c = _grfconfig; c != NULL; c = c->next) {
|
|
if (c->grfid == grfid) return c;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
|
|
/* Build a space separated list of parameters, and terminate */
|
|
char *GRFBuildParamList(char *dst, const GRFConfig *c, const char *last)
|
|
{
|
|
uint i;
|
|
|
|
/* Return an empty string if there are no parameters */
|
|
if (c->num_params == 0) return strecpy(dst, "", last);
|
|
|
|
for (i = 0; i < c->num_params; i++) {
|
|
if (i > 0) dst = strecpy(dst, " ", last);
|
|
dst += snprintf(dst, last - dst, "%d", c->param[i]);
|
|
}
|
|
return dst;
|
|
}
|
|
|
|
/** Base GRF ID for OpenTTD's base graphics GRFs. */
|
|
static const uint32 OPENTTD_GRAPHICS_BASE_GRF_ID = BSWAP32(0xFF4F5400);
|
|
|
|
/**
|
|
* Checks whether this GRF is a OpenTTD base graphic GRF.
|
|
* @return true if and only if it is a base GRF.
|
|
*/
|
|
bool GRFConfig::IsOpenTTDBaseGRF() const
|
|
{
|
|
return (this->grfid & 0x00FFFFFF) == OPENTTD_GRAPHICS_BASE_GRF_ID;
|
|
}
|
|
|
|
|
|
static const SaveLoad _grfconfig_desc[] = {
|
|
SLE_STR(GRFConfig, filename, SLE_STR, 0x40),
|
|
SLE_VAR(GRFConfig, grfid, SLE_UINT32),
|
|
SLE_ARR(GRFConfig, md5sum, SLE_UINT8, 16),
|
|
SLE_ARR(GRFConfig, param, SLE_UINT32, 0x80),
|
|
SLE_VAR(GRFConfig, num_params, SLE_UINT8),
|
|
SLE_END()
|
|
};
|
|
|
|
|
|
static void Save_NGRF()
|
|
{
|
|
int index = 0;
|
|
|
|
for (GRFConfig *c = _grfconfig; c != NULL; c = c->next) {
|
|
if (HasBit(c->flags, GCF_STATIC)) continue;
|
|
SlSetArrayIndex(index++);
|
|
SlObject(c, _grfconfig_desc);
|
|
}
|
|
}
|
|
|
|
|
|
static void Load_NGRF()
|
|
{
|
|
ClearGRFConfigList(&_grfconfig);
|
|
while (SlIterateArray() != -1) {
|
|
GRFConfig *c = CallocT<GRFConfig>(1);
|
|
SlObject(c, _grfconfig_desc);
|
|
AppendToGRFConfigList(&_grfconfig, c);
|
|
}
|
|
|
|
/* Append static NewGRF configuration */
|
|
AppendStaticGRFConfigs(&_grfconfig);
|
|
}
|
|
|
|
extern const ChunkHandler _newgrf_chunk_handlers[] = {
|
|
{ 'NGRF', Save_NGRF, Load_NGRF, CH_ARRAY | CH_LAST }
|
|
};
|
|
|
|
|
|
|